diff options
author | Josh Chen | 2020-05-24 20:44:33 +0200 |
---|---|---|
committer | Josh Chen | 2020-05-24 20:44:33 +0200 |
commit | 3ad2f03f7dbc922e2c711241146db091d193003d (patch) | |
tree | 6db26730510b119bf10dc836d21a9869ace6c416 /spartan | |
parent | 6f37e6b72905eff8f2c7078823e5577bc5b55eb0 (diff) |
new work on elimination tactic
Diffstat (limited to '')
-rw-r--r-- | spartan/lib/elimination.ML | 38 | ||||
-rw-r--r-- | spartan/lib/tactics.ML | 81 | ||||
-rw-r--r-- | spartan/theories/Equivalence.thy | 2 | ||||
-rw-r--r-- | spartan/theories/Identity.thy | 2 | ||||
-rw-r--r-- | spartan/theories/Spartan.thy | 32 |
5 files changed, 113 insertions, 42 deletions
diff --git a/spartan/lib/elimination.ML b/spartan/lib/elimination.ML index bd23c94..617f83e 100644 --- a/spartan/lib/elimination.ML +++ b/spartan/lib/elimination.ML @@ -1,38 +1,46 @@ (* Title: elimination.ML Author: Joshua Chen -Type elimination automation. +Type elimination setup. *) -structure Elim = struct +structure Elim: sig + +val rules: Proof.context -> (thm * indexname list) list +val lookup_rule: Proof.context -> Termtab.key -> (thm * indexname list) option +val register_rule: term list -> thm -> Context.generic -> Context.generic + +end = struct (** Context data **) + (* Elimination rule data *) + +(*Stores elimination rules together with a list of the indexnames of the + variables each rule eliminates. Keyed by head of the type being eliminated.*) structure Rules = Generic_Data ( - type T = thm Termtab.table + type T = (thm * indexname list) Termtab.table val empty = Termtab.empty val extend = I - val merge = Termtab.merge Thm.eq_thm_prop + val merge = Termtab.merge (eq_fst Thm.eq_thm_prop) ) -fun register_rule rl = +fun rules ctxt = map (op #2) (Termtab.dest (Rules.get (Context.Proof ctxt))) +fun lookup_rule ctxt = Termtab.lookup (Rules.get (Context.Proof ctxt)) +fun register_rule tms rl = let val hd = Term.head_of (Lib.type_of_typing (Thm.major_prem_of rl)) - in Rules.map (Termtab.update (hd, rl)) end + in Rules.map (Termtab.update (hd, (rl, map (#1 o dest_Var) tms))) end -fun get_rule ctxt = Termtab.lookup (Rules.get (Context.Proof ctxt)) -fun rules ctxt = map (op #2) (Termtab.dest (Rules.get (Context.Proof ctxt))) - -(* Set up [elims] attribute *) +(* [elims] attribute *) val _ = Theory.setup ( Attrib.setup \<^binding>\<open>elims\<close> - (Scan.lift (Scan.succeed (Thm.declaration_attribute register_rule))) "" + (Scan.repeat Args.term_pattern >> + (Thm.declaration_attribute o register_rule)) + "" #> Global_Theory.add_thms_dynamic (\<^binding>\<open>elims\<close>, - fn context => (rules (Context.proof_of context))) + fn context => (map #1 (rules (Context.proof_of context)))) ) -(** General elimination-induction tactic **) - - end diff --git a/spartan/lib/tactics.ML b/spartan/lib/tactics.ML index c72947a..024abc1 100644 --- a/spartan/lib/tactics.ML +++ b/spartan/lib/tactics.ML @@ -4,7 +4,7 @@ General tactics for dependent type theory. *) -structure Tactics: +structure Tactics(* : sig val assumptions_tac: Proof.context -> int -> tactic @@ -18,7 +18,7 @@ val intro_tac: Proof.context -> int -> tactic val intros_tac: Proof.context -> int -> tactic val elims_tac: term option -> Proof.context -> int -> tactic -end = struct +end *) = struct (*An assumption tactic that only solves typing goals with rigid terms and judgmental equalities without schematic variables*) @@ -56,25 +56,22 @@ fun typechk_tac ctxt = let val net = Tactic.build_net ((Named_Theorems.get ctxt \<^named_theorems>\<open>typechk\<close>) @(Named_Theorems.get ctxt \<^named_theorems>\<open>intros\<close>) - @(Elim.rules ctxt)) + @(map #1 (Elim.rules ctxt))) in (resolve_from_net_tac ctxt net) i end else no_tac) in - CHANGED o REPEAT o REPEAT_ALL_NEW (known_tac ctxt ORELSE' tac) + REPEAT_ALL_NEW (known_tac ctxt ORELSE' tac) end (*Many methods try to automatically discharge side conditions by typechecking. Switch this flag off to discharge by non-unifying assumption instead.*) val auto_typechk = Attrib.setup_config_bool \<^binding>\<open>auto_typechk\<close> (K true) +fun side_cond_tac ctxt = CHANGED o REPEAT o + (if Config.get ctxt auto_typechk then typechk_tac ctxt else known_tac ctxt) + (*Combinator runs tactic and tries to discharge all new typing side conditions*) -fun SIDE_CONDS tac ctxt = - let - val side_cond_tac = - if Config.get ctxt auto_typechk then typechk_tac ctxt else known_tac ctxt - in - tac THEN_ALL_NEW (TRY o side_cond_tac) - end +fun SIDE_CONDS tac ctxt = tac THEN_ALL_NEW (TRY o side_cond_tac ctxt) local fun mk_rules _ ths [] = ths @@ -87,13 +84,11 @@ fun mk_rules _ ths [] = ths in (*Resolves with given rules, discharging as many side conditions as possible*) -fun rule_tac ths ctxt = - SIDE_CONDS (resolve_tac ctxt (mk_rules 0 [] ths)) ctxt +fun rule_tac ths ctxt = resolve_tac ctxt (mk_rules 0 [] ths) (*Attempts destruct-resolution with the n-th premise of the given rules*) -fun dest_tac opt_n ths ctxt = SIDE_CONDS (dresolve_tac ctxt - (mk_rules (case opt_n of NONE => 0 | SOME 0 => 0 | SOME n => n-1) [] ths)) - ctxt +fun dest_tac opt_n ths ctxt = dresolve_tac ctxt + (mk_rules (case opt_n of NONE => 0 | SOME 0 => 0 | SOME n => n-1) [] ths) end @@ -107,17 +102,17 @@ fun intros_tac ctxt = SUBGOAL (fn (_, i) => (*Basic elimination tactic, only uses existing type judgments from the context (performs no type synthesis)*) fun elims_tac opt_tm ctxt = case opt_tm of - NONE => SUBGOAL (fn (_, i) => eresolve_tac ctxt (Elim.rules ctxt) i) + NONE => SUBGOAL (fn (_, i) => eresolve_tac ctxt (map #1 (Elim.rules ctxt)) i) | SOME tm => SUBGOAL (fn (goal, i) => let fun elim_rule typing = let val hd = head_of (Lib.type_of_typing typing) - val opt_rl = Elim.get_rule ctxt hd + val opt_rl = Elim.lookup_rule ctxt hd in case opt_rl of NONE => Drule.dummy_thm - | SOME rl => Drule.infer_instantiate' ctxt + | SOME (rl, _) => Drule.infer_instantiate' ctxt [SOME (Thm.cterm_of ctxt tm)] rl end @@ -137,6 +132,54 @@ fun elims_tac opt_tm ctxt = case opt_tm of ORELSE' eresolve_tac ctxt candidate_rules') i end) +(* Revamped general induction/elimination *) + +(*Pushes a context/goal premise typing t:T into a \<Prod>-type*) +fun internalize_fact_tac t = + Subgoal.FOCUS_PARAMS (fn {context = ctxt, concl = raw_concl, ...} => + let + val concl = Logic.strip_assums_concl (Thm.term_of raw_concl) + val C = Lib.type_of_typing concl + val B = Thm.cterm_of ctxt (Lib.lambda_var t C) + val a = Thm.cterm_of ctxt t + (*The resolvent is PiE[where ?B=B and ?a=a]*) + val resolvent = + Drule.infer_instantiate' ctxt [NONE, NONE, SOME B, SOME a] @{thm PiE} + in + HEADGOAL (resolve_tac ctxt [resolvent]) + (*known_tac infers the correct type T inferred by unification*) + THEN SOMEGOAL (known_tac ctxt) + end) + +fun elim_tac' tms ctxt = case tms of + [] => SUBGOAL (fn (_, i) => eresolve_tac ctxt (map #1 (Elim.rules ctxt)) i) + | major::_ => SUBGOAL (fn (goal, i) => + let + val template = Lib.typing_of_term major + val facts = Proof_Context.facts_of ctxt + val prems = Logic.strip_assums_hyp goal + val types = + map (Thm.prop_of o #1) (Facts.could_unify facts template) + @ filter (fn prem => Term.could_unify (template, prem)) prems + |> map Lib.type_of_typing + in case types of + [] => no_tac + | _ => + let + val rule_insts = map ((Elim.lookup_rule ctxt) o Term.head_of) types + val rules = flat (map + (fn rule_inst => case rule_inst of + NONE => [] + | SOME (rl, idxnames) => [Drule.infer_instantiate ctxt + (idxnames ~~ map (Thm.cterm_of ctxt) tms) rl]) + rule_insts) + in + resolve_tac ctxt rules i + THEN RANGE (replicate (length tms) (typechk_tac ctxt)) 1 + end handle Option => no_tac + end) + + end open Tactics diff --git a/spartan/theories/Equivalence.thy b/spartan/theories/Equivalence.thy index 97cd18d..73f1e0d 100644 --- a/spartan/theories/Equivalence.thy +++ b/spartan/theories/Equivalence.thy @@ -276,7 +276,7 @@ Lemma (derive) biinv_imp_qinv: ultimately have "g ~ h" apply (rewrite to "g \<circ> (id B)" id_right[symmetric]) apply (rewrite to "(id A) \<circ> h" id_left[symmetric]) - by (rule homotopy_transitive) (assumption, typechk) + by (rule homotopy_transitive) (assumption+, typechk) then have "f \<circ> g ~ f \<circ> h" by (rule homotopy_funcomp_right) diff --git a/spartan/theories/Identity.thy b/spartan/theories/Identity.thy index 4a4cc40..3f16f4c 100644 --- a/spartan/theories/Identity.thy +++ b/spartan/theories/Identity.thy @@ -40,7 +40,7 @@ axiomatization where lemmas [intros] = IdF IdI and - [elims] = IdE and + [elims "?p" (*"?x" "?y"*)] = IdE and [comps] = Id_comp diff --git a/spartan/theories/Spartan.thy b/spartan/theories/Spartan.thy index 5274ea2..ab2606e 100644 --- a/spartan/theories/Spartan.thy +++ b/spartan/theories/Spartan.thy @@ -160,10 +160,25 @@ ML_file \<open>../lib/elimination.ML\<close> \<comment> \<open>declares the [eli named_theorems intros and comps lemmas [intros] = PiF PiI SigF SigI and - [elims] = PiE SigE and + [elims "?f"] = PiE and + [elims "?p"] = SigE and [comps] = beta Sig_comp and [cong] = Pi_cong lam_cong Sig_cong +ML \<open> +val ctxt = @{context}; +val typing = @{term "t: \<Sum>x: A. B x"}; +val insts = []; +let + val (eterm, typing) = Lib.dest_typing typing + val (rl, tms) = the (Elim.lookup_rule ctxt (Term.head_of typing)) +in + Drule.infer_instantiate @{context} + (tms ~~ map (Thm.cterm_of ctxt) (eterm::insts)) + rl +end +\<close> + ML_file \<open>../lib/tactics.ML\<close> method_setup assumptions = @@ -180,25 +195,30 @@ method_setup intro = method_setup intros = \<open>Scan.succeed (fn ctxt => SIMPLE_METHOD (HEADGOAL (intros_tac ctxt)))\<close> -method_setup elim = +method_setup elim' = \<open>Scan.option Args.term >> (fn tm => fn ctxt => SIMPLE_METHOD' (SIDE_CONDS (elims_tac tm ctxt) ctxt))\<close> +method_setup elim = + \<open>Scan.repeat Args.term >> (fn tms => fn ctxt => + SIMPLE_METHOD' (SIDE_CONDS (elim_tac' tms ctxt) ctxt))\<close> + method elims = elim+ (*This could possibly use additional simplification hints via a simp: modifier*) method_setup typechk = - \<open>Scan.succeed (fn ctxt => SIMPLE_METHOD ( - CHANGED (ALLGOALS (TRY o typechk_tac ctxt))))\<close> + \<open>Scan.succeed (fn ctxt => SIMPLE_METHOD' ( + SIDE_CONDS (typechk_tac ctxt) ctxt)) + (* CHANGED (ALLGOALS (TRY o typechk_tac ctxt)))) *)\<close> method_setup rule = \<open>Attrib.thms >> (fn ths => fn ctxt => - SIMPLE_METHOD (HEADGOAL (rule_tac ths ctxt)))\<close> + SIMPLE_METHOD (HEADGOAL (SIDE_CONDS (rule_tac ths ctxt) ctxt)))\<close> method_setup dest = \<open>Scan.lift (Scan.option (Args.parens Parse.int)) -- Attrib.thms >> (fn (opt_n, ths) => fn ctxt => - SIMPLE_METHOD (HEADGOAL (dest_tac opt_n ths ctxt)))\<close> + SIMPLE_METHOD (HEADGOAL (SIDE_CONDS (dest_tac opt_n ths ctxt) ctxt)))\<close> subsection \<open>Rewriting\<close> |