reflect
-predicate. Multi-rewrite rulesAuthor: | Anton Trunov |
---|---|
Date: | April 22, 2021 |
Set Implicit Arguments. Unset Strict Implicit. Unset Printing Implicit Defensive.
reflect
-predicateAs we mentioned earlier, the SSReflect proof methodology is based on having logical and symbolic (e.g. boolean) representations intermixed in a goal, so that the user can switch between those to drive the proof process.
The SSReflect methodology, thus, favors computable predicates and relations.
Let's see an example now.
T:Typep:pred Ts:seq Tall p s -> [seq x <- s | p x] = sT:Typep:pred Ts:seq Tall p s -> [seq x <- s | p x] = s
First of all, let's try and understand the goal we see here. There are several things:
p : pred T
means p
is a computable predicate
over type T
, i.e. pred T
is T -> bool
;all p s
means all elements of sequence s
satisfy predicate p
;[seq x <- s | p x]
notation stands for
filter (fun x => p x) s
.We can check all the above bullet points with the following three simple commands:
But wait a minute, how come we can have a term
of type bool
as an assumption in our goal?
T:Typep:pred Ts:seq T
all p s -> [seq x <- s | p x] = s
And why the following works at all?
T:Typep:pred Ts:seq T
all p s -> [seq x <- s | p x] = s
This works because of the mechanism of implicit coercions Coq uses to make sense of goals like the current one. To see clearly what is going on here, use the following vernacular:
T:Typep:pred Ts:seq T
is_true (all p s) -> [seq x <- s | p x] = s
Now we can see Coq inserted the corcion
is_true
around all p s
so the goal would
actually make sense.
If eq^~ true
does not really make sense for
you:
Another approach would be to unfold is_true
:
T:Typep:pred Ts:seq T
all p s = true -> [seq x <- s | p x] = s
Given the above, is_true b
means b = true
-- this is one way to embed booleans into Prop
.
T:Typep:pred Ts:seq T
all p s = true -> [seq x <- s | p x] = s
Now that we understand the goal we can prove
it by induction on s
:
T:Typep:pred Tx:Ts:seq TIHs:all p s = true -> [seq x <- s | p x] = s
p x && all p s = true ->
(if p x
then x :: [seq x <- s | p x]
else [seq x <- s | p x]) = x :: s
In our top assumption we have both p x
which
is needed to simplify the if
-expression in the
goal conclusion and all p s
which is needed to
apply the induction hypothesis. So, at this point
we need to transform an assumption from its
symbolic form ?x && ?y = true
into its logical
form ?x = true /\ ?y = true
.
To do this we are going to use the mechanism of views supported by SSReflect.
T:Typep:pred Tx:Ts:seq TIHs:all p s = true -> [seq x <- s | p x] = s
p x /\ all p s ->
(if p x
then x :: [seq x <- s | p x]
else [seq x <- s | p x]) = x :: s
We already know the /lemma
syntax as we saw
how it works for lemmas which are implications. In
this case it's something like this as well,
although there is some implicit machinery at work
too. The mechanism is called view hints and we
are going to cover it later. The rest of the proof
is very straightforward.
T:Typep:pred Tx:Ts:seq TIHs:all p s = true -> [seq x <- s | p x] = sp x -> all p s -> (if p x then x :: [seq x <- s | p x] else [seq x <- s | p x]) = x :: sT:Typep:pred Tx:Ts:seq TIHs:all p s = true -> [seq x <- s | p x] = sall p s -> x :: [seq x <- s | p x] = x :: sT:Typep:pred Tx:Ts:seq TIHs:all p s = true -> [seq x <- s | p x] = s[seq x <- s | p x] = s -> x :: [seq x <- s | p x] = x :: sdone.T:Typep:pred Tx:Ts:seq TIHs:all p s = true -> [seq x <- s | p x] = sx :: s = x :: s
Let us refactor the proof above into something more idiomatic
by elim: s=> //= x s IHs /andP[-> /IHs->]. Qed.T:Typep:pred Ts:seq Tall p s -> [seq x <- s | p x] = s
reflect
-predicate: definitionSo, what is reflect
in the type of andP
from above?
The reflect
type family (or indexed type, in
other words) connects a decidable proposition to
the corresponding boolean expression. This is not
the first time we come across an indexed type but
it's the first time we run into an indexed type
with two constructors where we can clearly see the
difference between parameters (which have to be
fixed across constructors, P
here is a
parameter) and indices (which can vary between
constructors). In this case, type-level boolean
term indicates which constructor has been used to
construct a term of type reflect P b
, e.g. when
one has a term of type reflect P true
they know
the ReflectT
constructor has been used to
construct the term which means we have a proof of
P
we can get by pattern-matching on a term of
type reflect P true
. (And pattern-matching on a
term of type reflect P false
yields a refutation
of P
.)
forall P : Prop, reflect P true -> Pexact: ( fun r : reflect P true => match r in reflect _ b return (b = true -> P) with | ReflectT p => fun E => p | ReflectF np => fun E => ltac:(done) end erefl).P:Propreflect P true -> PP:Propreflect P true -> PP:PropP -> PP:Prop~ P -> P
case
actually leads to unprovable goals
here. It's not intelligent enough to solve this
goal which requires dependent pattern matching.
We need to use a special version of case
for
type families. Here is the syntax we need to solve
this.
P:Propreflect P true -> PP:PropR:reflect P truePP:Prop_p_:PE:true = truePP:Prop_n_:~ PE:true = falseP
To the right of /
we put a term (R
in this
case) we are going to case analyse -- this needs
to be a type family. And to the left of /
we put
the indices of the type family we'd like to get
substituted. Notice that we need to keep the
relation between the original value of the index
true
and its values inside the branches of the
underlying match
-expression and this is what E
in case: E
is for.
P:Prop_p_:PE:true = truePP:Prop_n_:~ PE:true = falsePdone.P:Prop_n_:~ PE:true = falsePforall P : Prop, reflect P true -> P
Instead of move=> R; case E: true / R
we can
use a shorhand:
by move=> P; case E: true /.forall P : Prop, reflect P true -> P
Moreover, SSReflect can figure out the indices on its own:
by move=> P; case E: _/. Qed.
A lemma like reflect P false -> ~ P
can be
proved analogously.
To reinforce what has been said already, let
us formally prove several lemmas about reflect
.
For instance, we can show P
if and only if b =
true
as two separate lemmas.
P:Propb:boolreflect P b -> P -> bP:Propb:boolreflect P b -> P -> bP:Propb:boolP -> P -> trueP:Propb:bool~ P -> P -> false
The index b
here works as a rewrite rule.
P:Propb:boolP -> P -> trueP:Propb:bool~ P -> P -> falseby move=> /[apply]. Qed.P:Propb:bool~ P -> P -> false
The other direction is left as an exercise for the reader.
Admitted.P:Propb:boolreflect P b -> b -> P
Essentially, we have shown reflect P b -> (b
<-> P)
, i.e. reflect P b
connects a decidable
proposition P
to its decision procedure (the
boolean expression b
).
For example, we can show reflect P b
lets us
use classical reasoning (exercise):
Admitted.P:Propb:boolreflect P b -> P \/ ~ P
Lets look at how to build reflect
-predicates
for a couple of standard connectives.
b, c:boolreflect (b /\ c) (b && c)b, c:boolreflect (b /\ c) (b && c)c:boolreflect (true /\ c) (true && c)c:boolreflect (false /\ c) (false && c)c:boolreflect (true /\ c) (true && c)c:boolreflect (false /\ c) (false && c)reflect (true /\ true) (true && true)reflect (true /\ false) (true && false)c:boolreflect (false /\ c) (false && c)reflect (true /\ true) (true && true)reflect (true /\ false) (true && false)c:boolreflect (false /\ c) (false && c)true /\ truereflect (true /\ false) (true && false)c:boolreflect (false /\ c) (false && c)reflect (true /\ false) (true && false)c:boolreflect (false /\ c) (false && c)reflect (true /\ false) (true && false)c:boolreflect (false /\ c) (false && c)~ (true /\ false)c:boolreflect (false /\ c) (false && c)c:boolreflect (false /\ c) (false && c)c:boolreflect (false /\ c) (false && c)c:bool~ (false /\ c)b, c:boolreflect (b /\ c) (b && c)
An idiomatic solution would look something like so:
by case: b; case: c; constructor=> //; case. Qed.b, c:boolreflect (~~ b \/ ~~ c) (~~ (b && c))b, c:boolreflect (~~ b \/ ~~ c) (~~ (b && c))~ (~~ true \/ ~~ true)~~ true \/ ~~ false~~ false \/ ~~ true~~ false \/ ~~ false~ (~~ true \/ ~~ true)~~ true \/ ~~ false~~ false \/ ~~ true~~ false \/ ~~ false~~ true \/ ~~ false~~ false \/ ~~ true~~ false \/ ~~ false~~ true \/ ~~ false~~ false \/ ~~ true~~ false \/ ~~ false~~ false \/ ~~ true~~ false \/ ~~ false~~ false \/ ~~ true~~ false \/ ~~ falseby left.~~ false \/ ~~ falseb, c:boolreflect (~~ b \/ ~~ c) (~~ (b && c))
This seems to be a rare opportunity for an
automatic tactic to fill in the blanks for us: the
intuition
tactic can take care of the subgoals
generated by our brute force approach.
by case: b; case: c; constructor; intuition. Qed.
Sometimes we will need to use reflect b b
as
a placeholder. This is an easy exercise.
Admitted.b:boolreflect b b
Let's continue figuring out how and why andP
works.
b, c:boolb && c -> b /\ cb, c:boolb && c -> b /\ cb, c:boolb /\ c -> b /\ cb, c:boolb /\ c -> b /\ c
We see andP
in a context of elimTF
, let's
see what it is.
b, c:bool
b /\ c -> b /\ c
elimTF
is a generalization of elimT
we proved above.
b, c:bool
b && c -> b /\ c
Let us see what is going on when we say
move/andP
:
b, c:boolHb:b && cb /\ cb, c:boolHb:b && cb /\ cb, c:boolb && c -> b /\ cb, c:boolb /\ c -> b /\ c
But where elimTF
comes from? SSReflect uses
the mechanism of view hints which provide some
wrapping lemmas such as elimTF
above. The user
can add a new view hint using the Hint View
vernacular Hint View for move/ elimTF|3
.
Here elimTF
is declared as a view hint for
the move/
command. The (optional) number 3
specifies the number of implicit arguments
to be considered for the declared hint view lemma.
The ssrbool.v
module already declares a
numbers of view hints, so adding new ones should
be justified. For instance, one might need to do
it if one defines a new logical connective.
exact: id. Qed.
But why is elimTF
's type so complex? Because
it's applicable not only in the case the goal's
top assumtion is of the form b && c
, but it also
works for b && c = false
.
b, c:boolb && c = false -> ~ (b /\ c)b, c:boolb && c = false -> ~ (b /\ c)b, c:bool~ (b /\ c) -> ~ (b /\ c)b, c:bool~ (b /\ c) -> ~ (b /\ c)b, c:boolb && c = false -> ~ (b /\ c)b, c:boolHb:b && c = false~ (b /\ c)exact: @elimTF (b /\ c) (b && c) false (@andP b c) Hb. Qed.b, c:boolHb:b && c = false~ (b /\ c)
Reflection views usually work in both directions
b, c:boolb /\ c -> b && cb, c:boolb /\ c -> b && cb, c:boolb && c -> b && cb, c:boolb && c -> b && cb, c:boolb && c -> b && c
introT
view hint gets implicitly inserted
because it is also declared with Hint View
command.
done. Qed.
b, c:boolb /\ c -> b && cb, c:boolb /\ c -> b && cb, c:boolbc:b /\ cb && c
If we'd like, instead of the assumption,
change the goal, we can use apply/
tactic:
b, c:boolbc:b /\ cb /\ cb, c:boolbc:b /\ cb /\ c
In this case the introTF
view hint gets
inserted because the ssrbool
module introduces
the correspoding view hint: Hint View for apply/ introTF|3
done. Qed.b, c:boolbc:b /\ cb /\ c
Again, introTF
is a general lemma to
accomodate goals of the form _ = false
.
b, c:bool~ (b /\ c) -> b && c = falseb, c:bool~ (b /\ c) -> b && c = falseb, c:boolab:~ (b /\ c)b && c = falseb, c:boolab:~ (b /\ c)~ (b /\ c)b, c:boolab:~ (b /\ c)~ (b /\ c)done. Qed.b, c:boolab:~ (b /\ c)~ (b /\ c)
Let's look at some more machinery to work with
specifiations for decision procedures. We are
going to look at the eqn
-- the decision
procedure for equality on the nat
type.
n, m:natreflect (n = m) (eqn n m)n, m:natreflect (n = m) (eqn n m)
One way to prove this is to turn [reflect] into a bi-implication and prove the two directions by induction separately. An idiomatic way to do that is as follows:
n, m:nateqn n m -> n = mn, m:natn = m -> eqn n mn, m:nateqn n m -> n = mn, m:natn = m -> eqn n m
We need the trivial reflection view idP
here
to convert the boolean expression eqn n m
to the
proposition eqn n m = true
(you don't see the =
true
part in the subgoals because of the
is_true
coercion).
The rest of the proof should be trivial at this point.
n, m:nateqn n m -> n = mn, m:natn = m -> eqn n mby move=> ->; elim: m. Qed.n, m:natn = m -> eqn n m
Here is an example of using iffP
with a
non-idP
argument. Here we use eqType
-- a type
with decidable equality and some machinery
associated with it, like eqP
.
T:eqTypen:natx, y:Treflect (y = x /\ 0 < n) (y \in nseq n x)T:eqTypen:natx, y:Treflect (y = x /\ 0 < n) (y \in nseq n x)T:eqTypen:natx, y:Treflect (y = x /\ 0 < n) ((0 < n) && (y == x))T:eqTypen:natx, y:Treflect (y = x /\ 0 < n) ((y == x) && (0 < n))T:eqTypen:natx, y:Ty == x /\ 0 < n -> y = x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:Ty == x /\ 0 < n -> y = x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:Ty == x -> 0 < n -> y = x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:Ty = x -> 0 < n -> y = x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:Ty = x -> 0 < n -> y = x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:T0 < n -> x = x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:Ty = x /\ 0 < n -> y == x /\ 0 < nT:eqTypen:natx, y:Tx == x /\ truedone.T:eqTypen:natx, y:Ttrue /\ true
A more idiomatic solution
T:eqTypen:natx, y:Treflect (y = x /\ 0 < n) (y \in nseq n x)T:eqTypen:natx, y:Ty = x -> 0 < n -> y = x /\ 0 < nT:eqTypen:natx, y:Ty == x -> 0 < n -> y == x /\ 0 < n
There is some code duplication here which can
be reduced using -[ ]
syntax:
by rewrite mem_nseq andbC; apply: (iffP andP)=> -[/eqP].T:eqTypen:natx, y:Treflect (y = x /\ 0 < n) (y \in nseq n x)
We cannot say just [/eqP]
because Coq
expects us to provide tactics for both subgoals
and not just one. To bypass this restriction we
use the -
syntax.
Qed.
reflect
predicatesOne can rewrite with view lemmas if those
represent equations, like maxn_idPl
.
Let's see an example:
m, n1, n2:nat(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:nat(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= n1) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)
Why does this trick work?
m, n1, n2:natle_n21:n2 <= n1
(m <= n1) = (m <= n1) || (m <= n2)
m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)
OK, this is an ordinary equation, no wonder
rewrite
works. The view lemma [maxn_idPl] is
not a function but behaves like one here. Let us
check coercions!
m, n1, n2:natle_n21:is_true (n2 <= n1)(m <= n1) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:is_true (n1 <= n2)(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:is_true (n2 <= n1)(m <= n1) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:is_true (n1 <= n2)(m <= maxn n1 n2) = (m <= n1) || (m <= n2)
No magic: elimT
get implicitly inserted.
m, n1, n2:natle_n21:n2 <= n1(m <= n1) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= n1) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)
elimT
is a coercion from reflect
to
Funclass
, which means it gets inserted when one
uses a view lemma as a function.
Essentially we invoke the following tactic:
m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= n1) = (m <= n1) || (m <= n2)m, n1, n2:natle_n12:n1 <= n2(m <= maxn n1 n2) = (m <= n1) || (m <= n2)
One can easily finish the proof, but let's simplify it first.
m, n1, n2:nat
(m <= maxn n1 n2) = (m <= n1) || (m <= n2)
We remove the symmetrical case first:
m, n1, n2:nat(forall n1 n2 : nat, n2 <= n1 -> (m <= maxn n1 n2) = (m <= n1) || (m <= n2)) -> (m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:nat(forall n1 n2 : nat, n2 <= n1 -> (m <= maxn n1 n2) = (m <= n1) || (m <= n2)) -> (m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= maxn n1 n2) = (m <= n1) || (m <= n2)m, n1, n2:natle_n21:n2 <= n1(m <= n1) = (m <= n1) || (m <= n2)
The rest is an exercise
Abort.
Check out some other specs in the seq
and
ssrnat
modules!
The inside
syntax lets one look for lemmas
inside a particular module only.
We have seen reflect
-predicates being able
to rewrite boolean expressions corresponding to
its index. We can take this approach even further.
Let's see an example of a mutli-rule.
m, n:natm <= n <= m -> (m == n) || (n < m) || (m + n == 0)by case: ltngtP.m, n:natm <= n <= m -> (m == n) || (n < m) || (m + n == 0)
That was quick!
m, n:natm <= n <= m -> (m == n) || (n < m) || (m + n == 0)
case: ltngtP
performs case analysis: it
generates three subgoals corresponding to thee
three cases, strictly less, equal, strictly
greater.
m, n:nat
n < m ->
false && true -> false || true || (m + n == 0)
m, n:natm < n ->
true && false -> false || false || (m + n == 0)
m, n:natn = m -> true && true -> true || false || (m + n == 0)
Notice that a lot of boolean expressions got
replaced with true
or false
depending on a
concrete case we are covering. The ltngtP
also
works as a rewrite rule.
m, n:natn < m -> false && true -> false || true || (m + n == 0)m, n:natm < n -> true && false -> false || false || (m + n == 0)m, n:natn = m -> true && true -> true || false || (m + n == 0)m, n:natm < n -> true && false -> false || false || (m + n == 0)m, n:natn = m -> true && true -> true || false || (m + n == 0)m, n:natm < n -> true && false -> false || false || (m + n == 0)m, n:natn = m -> true && true -> true || false || (m + n == 0)m, n:natn = m -> true && true -> true || false || (m + n == 0)done. Qed.m, n:natn = m -> true -> true
Let's see how one can implement this combined lemma.
Module Trichotomy.
First, we define a type family (indexed type)
with indices corresponding to expressions we want
to rewrite in subgoals. We use the Variant
vernacular here which is exactly like Inductive
but the type cannot refer to itself in its
definition, in other words it's a non-recursive
inductive type.
Variant compare_nat m n :
bool -> bool -> bool -> bool -> bool -> bool -> Type :=
| CompareNatLt of m < n :
compare_nat m n false false false true false true
| CompareNatGt of m > n :
compare_nat m n false false true false true false
| CompareNatEq of m = n :
compare_nat m n true true true true false false.
Next, we define a specification lemma which connect the type family above with concrete expressions we want to rewrite.
m, n:natcompare_nat m n (n == m) (m == n) (n <= m) (m <= n) (n < m) (m < n)m, n:natcompare_nat m n (n == m) (m == n) (n <= m) (m <= n) (n < m) (m < n)m, n:natnm:m < ncompare_nat m n (n == m) (n == m) false (m <= n) ((n != m) && false) ((n != m) && (m <= n))m, n:natn <= m -> compare_nat m n (n == m) (n == m) true (m <= n) ((n != m) && true) ((n != m) && (m <= n))m, n:natnm:m < ncompare_nat m n (n == m) (n == m) false (m <= n) ((n != m) && false) ((n != m) && (m <= n))m, n:natn <= m -> compare_nat m n (n == m) (n == m) true (m <= n) ((n != m) && true) ((n != m) && (m <= n))m, n:natn <= m -> compare_nat m n (n == m) (n == m) true (m <= n) ((n != m) && true) ((n != m) && (m <= n))m, n:natlt_mn:n < meq_nm:truecompare_nat m n (n == m) (n == m) true false ((n != m) && true) ((n != m) && false)m, n:natlt_mn:m <= neq_nm:n == mcompare_nat m n (n == m) (n == m) true true ((n != m) && true) ((n != m) && true)m, n:natlt_mn:n < meq_nm:truecompare_nat m n (n == m) (n == m) true false ((n != m) && true) ((n != m) && false)m, n:natlt_mn:m <= neq_nm:n == mcompare_nat m n (n == m) (n == m) true true ((n != m) && true) ((n != m) && true)by rewrite eq_nm; constructor; apply/esym/eqP. Qed. End Trichotomy.m, n:natlt_mn:m <= neq_nm:n == mcompare_nat m n (n == m) (n == m) true true ((n != m) && true) ((n != m) && true)
We can take this proof apart during a seminar but here are things to observe here:
- The
rewrite
tactic support a language of patterns to focus rewriting on a particular part of the goal, e.g. in this case rewrite will only happen in a subterm corresponding to the_ == n
pattern.- The
ltnP
spec lemma which is likeltngtP
but simpler.- Just like one can do chained transformations of the top of the goal stack, one can do it for the conclusion:
apply/esym/eqP
.
Set Printing Coercions
: turn on printing of
implicit coercions -- very useful to better
understand goals.Search <pattern> inside <module>
: narrow the
scope of searching to a particular module
.-
action: can be used to connect unrelated
views (move=> /V1 - /V2
) or to force the
interpretation of []
as case splitting when
multiple subgoals are generated; -[/eqP]
.case ident: term
: case analyse on term
and
keep the corresponding equation in the context
under the name of ident
.case: index_pattern / type_family
: case
analyse on a term of indexed type and perform
substitutions of its indices according to
index_pattern
.constructor
: try to pick a data constructor
automatically.intuition
: a solver for propositional
intuitionistic logic.without loss
or wlog
: "without loss of
generality"-style reasoning.apply/view_lemma
: use the view mechanism on
the conclusion of the goal. Can be chained
together like apply/VL1/VL2/VL3
.rewrite
tactic supports patterns: rewrite
[<pattern>]<equation>
.