Skip to content
Snippets Groups Projects
Commit 8dd38309 authored by Ralf Jung's avatar Ralf Jung
Browse files

notypeclasses apply: do not unfold the type

also improve the error message
parent de2bf2a5
No related branches found
No related tags found
No related merge requests found
...@@ -628,28 +628,29 @@ Ltac opose_core p tac := ...@@ -628,28 +628,29 @@ Ltac opose_core p tac :=
(** Turn all leading ∀ and → of [p] into evars (∀-evars will be shelved), and (** Turn all leading ∀ and → of [p] into evars (∀-evars will be shelved), and
call [tac] with the term applied with those evars. This fill unfold definitions call [tac] with the term applied with those evars. This fill unfold definitions
to find leading ∀/→. to find leading ∀/→.
The type of [p] will be normalized by calling the [normalizer] function.
[_name_guard] is an unused argument where you can pass anything you want. If the [_name_guard] is an unused argument where you can pass anything you want. If the
argument is an intro pattern, those will be taken into account by the [fresh] argument is an intro pattern, those will be taken into account by the [fresh]
that is inside this tactic, avoiding name collisions that can otherwise arise. that is inside this tactic, avoiding name collisions that can otherwise arise.
This is a work-around for https://github.com/coq/coq/issues/18109. *) This is a work-around for https://github.com/coq/coq/issues/18109. *)
Ltac ospecialize_foralls p _name_guard tac := Ltac evar_foralls p _name_guard normalizer tac :=
let T := type of p in let T := type of p in
lazymatch eval hnf in T with lazymatch (normalizer T) with
| ?T1 ?T2 => | ?T1 ?T2 =>
(* This is the [fresh] where the presence of [_name_guard] matters. (* This is the [fresh] where the presence of [_name_guard] matters.
Note that the "opose_internal" is nice but not sufficient because Note that the "opose_internal" is nice but not sufficient because
it gets ignored when name mangling is enabled. *) it gets ignored when name mangling is enabled. *)
let pT1 := fresh "opose_internal" in let pT1 := fresh "__evar_foralls_internal" in
assert T1 as pT1; [| ospecialize_foralls (p pT1) _name_guard tac; clear pT1] assert T1 as pT1; [| evar_foralls (p pT1) _name_guard normalizer tac; clear pT1]
| x : ?T1, _ => | x : ?T1, _ =>
let e := mk_evar T1 in let e := mk_evar T1 in
ospecialize_foralls (p e) _name_guard tac evar_foralls (p e) _name_guard normalizer tac
| ?T1 => tac p | ?T1 => tac p
end. end.
Ltac opose_specialize_foralls_core p _name_guard tac := Ltac opose_specialize_foralls_core p _name_guard tac :=
opose_core p ltac:(fun p => ospecialize_foralls p _name_guard tac). opose_core p ltac:(fun p => evar_foralls p _name_guard ltac:(fun t => eval hnf in t) tac).
Tactic Notation "opose" "proof" uconstr(p) "as" simple_intropattern(pat) := Tactic Notation "opose" "proof" uconstr(p) "as" simple_intropattern(pat) :=
opose_core p ltac:(fun p => pose proof p as pat). opose_core p ltac:(fun p => pose proof p as pat).
...@@ -738,9 +739,16 @@ Tactic Notation "ospecialize" "*" uconstr(p) := ...@@ -738,9 +739,16 @@ Tactic Notation "ospecialize" "*" uconstr(p) :=
(** Tactic that works like [notypeclasses refine] but automatically determines (** Tactic that works like [notypeclasses refine] but automatically determines
the right number of [_]. This is *not* goal-directed, it will add [_] until the the right number of [_]. This is *not* goal-directed, it will add [_] until the
given term no longer has a function type (determined via [eval hnf]). *) given term no longer has a function type (determined without any unfolding). *)
Tactic Notation "notypeclasses" "apply" uconstr(p) := Tactic Notation "notypeclasses" "apply" uconstr(p) :=
opose_specialize_foralls_core p () ltac:(fun p => notypeclasses refine p). opose_core p ltac:(fun p => evar_foralls p () ltac:(fun t => t)
ltac:(fun p => notypeclasses refine p ||
let T := type of p in
match goal with |- ?G =>
fail "the given term has type" T "while it is expected to have type" G
end
)
).
(** The block definitions are taken from [Coq.Program.Equality] and can be used (** The block definitions are taken from [Coq.Program.Equality] and can be used
by tactics to separate their goal from hypotheses they generalize over. *) by tactics to separate their goal from hypotheses they generalize over. *)
......
...@@ -80,12 +80,7 @@ use opose proof instead. ...@@ -80,12 +80,7 @@ use opose proof instead.
"notypeclasses_apply_error" "notypeclasses_apply_error"
: string : string
The command has indeed failed with message: The command has indeed failed with message:
In environment Tactic failure: the given term has type Q while it is expected to have type
P, Q : Prop P.
H : P → Q
HQ : Q
opose_internal : P
The term "H opose_internal" has type "Q" while it is expected to have type
"P".
The command has indeed failed with message: The command has indeed failed with message:
No such assumption. No such assumption.
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment