Commit f0916dd9 authored by Adam Chlipala's avatar Adam Chlipala

Incorporate feedback from Nathan Collins

parent 30d492d9
...@@ -584,7 +584,7 @@ End tree. ...@@ -584,7 +584,7 @@ End tree.
Implicit Arguments Node [A n]. Implicit Arguments Node [A n].
(** We can redefine [sum] and [inc] for our new [tree] type. Again, it is useful to define a generic fold function first. This time, it takes in a function whose range is some [ffin] type, and it folds another function over the results of calling the first function at every possible [ffin] value. *) (** We can redefine [sum] and [inc] for our new [tree] type. Again, it is useful to define a generic fold function first. This time, it takes in a function whose domain is some [ffin] type, and it folds another function over the results of calling the first function at every possible [ffin] value. *)
Section rifoldr. Section rifoldr.
Variables A B : Set. Variables A B : Set.
......
...@@ -628,7 +628,7 @@ The term "pf" has type "a :: (ls1 ++ ls2) ++ ls3 = a :: ls1 ++ ls2 ++ ls3" ...@@ -628,7 +628,7 @@ The term "pf" has type "a :: (ls1 ++ ls2) ++ ls3 = a :: ls1 ++ ls2 ++ ls3"
end end
]] ]]
The conclusion has gotten markedly simpler. It seems counterintuitive that we can have an easier time of proving a more general theorem, but that is exactly the case here and for many other proofs that use dependent types heavily. Speaking informally, the reason why this kind of activity helps is that [match] annotations only support variables in certain positions. By reducing more elements of a goal to variables, built-in tactics can have more success building [match] terms under the hood. The conclusion has gotten markedly simpler. It seems counterintuitive that we can have an easier time of proving a more general theorem, but that is exactly the case here and for many other proofs that use dependent types heavily. Speaking informally, the reason why this kind of activity helps is that [match] annotations contain some positions where only variables are allowed. By reducing more elements of a goal to variables, built-in tactics can have more success building [match] terms under the hood.
In this case, it is helpful to generalize over our two proofs as well. *) In this case, it is helpful to generalize over our two proofs as well. *)
......
...@@ -26,7 +26,7 @@ Set Implicit Arguments. ...@@ -26,7 +26,7 @@ Set Implicit Arguments.
(** * Reifying Datatype Definitions *) (** * Reifying Datatype Definitions *)
(** The key to generic programming with dependent types is%\index{universe types}% _universe types_. This concept should not be confused with the idea of _universes_ from the metatheory of CIC and related languages. Rather, the idea of universe types is to define inductive types that provide _syntactic representations_ of Coq types. We cannot directly write CIC programs that do case analysis on types, but we _can_ case analyze on reified syntactic versions of those types. (** The key to generic programming with dependent types is%\index{universe types}% _universe types_. This concept should not be confused with the idea of _universes_ from the metatheory of CIC and related languages, which we will study in more detail in the next chapter. Rather, the idea of universe types is to define inductive types that provide _syntactic representations_ of Coq types. We cannot directly write CIC programs that do case analysis on types, but we _can_ case analyze on reified syntactic versions of those types.
Thus, to begin, we must define a syntactic representation of some class of datatypes. In this chapter, our running example will have to do with basic algebraic datatypes, of the kind found in ML and Haskell, but without additional bells and whistles like type parameters and mutually recursive definitions. Thus, to begin, we must define a syntactic representation of some class of datatypes. In this chapter, our running example will have to do with basic algebraic datatypes, of the kind found in ML and Haskell, but without additional bells and whistles like type parameters and mutually recursive definitions.
...@@ -138,7 +138,7 @@ Check hmake. ...@@ -138,7 +138,7 @@ Check hmake.
Definition size T dt (fx : fixDenote T dt) : T -> nat := Definition size T dt (fx : fixDenote T dt) : T -> nat :=
fx nat (hmake (B := constructorDenote nat) (fun _ _ r => foldr plus 1 r) dt). fx nat (hmake (B := constructorDenote nat) (fun _ _ r => foldr plus 1 r) dt).
(** Our definition is parameterized over a recursion scheme [fx]. We instantiate [fx] by passing it the function result type and a set of function cases, where we build the latter with [hmake]. The function argument to [hmake] takes three arguments: the representation of a constructor, its non-recursive arguments, and the results of recursive calls on all of its recursive arguments. We only need the recursive call results here, so we call them [r] and bind the other two inputs with wildcards. The actual case body is simple: we add together the recursive call results and increment the result by one (to account for the current constructor). This [foldr] function is an [hlist]-specific version defined in the [DepList] module. (** Our definition is parameterized over a recursion scheme [fx]. We instantiate [fx] by passing it the function result type and a set of function cases, where we build the latter with [hmake]. The function argument to [hmake] takes three arguments: the representation of a constructor, its non-recursive arguments, and the results of recursive calls on all of its recursive arguments. We only need the recursive call results here, so we call them [r] and bind the other two inputs with wildcards. The actual case body is simple: we add together the recursive call results and increment the result by one (to account for the current constructor). This [foldr] function is an [ilist]-specific version defined in the [DepList] module.
It is instructive to build [fixDenote] values for our example types and see what specialized [size] functions result from them. *) It is instructive to build [fixDenote] values for our example types and see what specialized [size] functions result from them. *)
...@@ -292,7 +292,7 @@ Eval compute in print (^ "true" (fun _ => "") ...@@ -292,7 +292,7 @@ Eval compute in print (^ "true" (fun _ => "")
::: HNil) bool_fix. ::: HNil) bool_fix.
(** %\vspace{-.15in}% [[ (** %\vspace{-.15in}% [[
= fun b : bool => if b then "true()" else "false()" = fun b : bool => if b then "true()" else "false()"
: bool -> s : bool -> string
]] ]]
*) *)
...@@ -656,7 +656,7 @@ Theorem map_id : forall T dt ...@@ -656,7 +656,7 @@ Theorem map_id : forall T dt
(* end thide *) (* end thide *)
(* end hide *) (* end hide *)
(** The base case is discharged automatically, and the inductive case looks like this, where [H] is the outer IH (for induction over [T] values) and [IHn] is the inner IH (for induction over the recursive arguments). (** The base case is discharged automatically, and the inductive case looks like this, where [H] is the outer IH (for induction over [T] values) and [IHr] is the inner IH (for induction over the recursive arguments).
[[ [[
H : forall i : fin (S n), H : forall i : fin (S n),
fx T fx T
......
...@@ -507,7 +507,7 @@ Check list_ind. ...@@ -507,7 +507,7 @@ Check list_ind.
forall l : list T, P l forall l : list T, P l
]] ]]
Thus, even though we just saw that [T] is added as an extra argument to the constructor [Cons], the inductive case in the type of [list_ind] (i.e., the third line of the type) includes no quantifier for [T], even though all of the other arguments are quantified explicitly. Parameters in other inductive definitions are treated similarly in stating induction principles. *) Thus, despite a very real sense in which the type [T] is an argument to the constructor [Cons], the inductive case in the type of [list_ind] (i.e., the third line of the type) includes no quantifier for [T], even though all of the other arguments are quantified explicitly. Parameters in other inductive definitions are treated similarly in stating induction principles. *)
(** * Mutually Inductive Types *) (** * Mutually Inductive Types *)
......
...@@ -582,13 +582,13 @@ Theorem and_True_conc : forall P Q, ...@@ -582,13 +582,13 @@ Theorem and_True_conc : forall P Q,
imp. imp.
Qed. Qed.
Theorem assoc_prem1 : forall P Q R S, Theorem pick_prem1 : forall P Q R S,
(P /\ (Q /\ R) --> S) (P /\ (Q /\ R) --> S)
-> ((P /\ Q) /\ R --> S). -> ((P /\ Q) /\ R --> S).
imp. imp.
Qed. Qed.
Theorem assoc_prem2 : forall P Q R S, Theorem pick_prem2 : forall P Q R S,
(Q /\ (P /\ R) --> S) (Q /\ (P /\ R) --> S)
-> ((P /\ Q) /\ R --> S). -> ((P /\ Q) /\ R --> S).
imp. imp.
...@@ -600,13 +600,13 @@ Theorem comm_prem : forall P Q R, ...@@ -600,13 +600,13 @@ Theorem comm_prem : forall P Q R,
imp. imp.
Qed. Qed.
Theorem assoc_conc1 : forall P Q R S, Theorem pick_conc1 : forall P Q R S,
(S --> P /\ (Q /\ R)) (S --> P /\ (Q /\ R))
-> (S --> (P /\ Q) /\ R). -> (S --> (P /\ Q) /\ R).
imp. imp.
Qed. Qed.
Theorem assoc_conc2 : forall P Q R S, Theorem pick_conc2 : forall P Q R S,
(S --> Q /\ (P /\ R)) (S --> Q /\ (P /\ R))
-> (S --> (P /\ Q) /\ R). -> (S --> (P /\ Q) /\ R).
imp. imp.
...@@ -626,8 +626,8 @@ Ltac search_prem tac := ...@@ -626,8 +626,8 @@ Ltac search_prem tac :=
|| (apply and_True_prem; tac) || (apply and_True_prem; tac)
|| match P with || match P with
| ?P1 /\ ?P2 => | ?P1 /\ ?P2 =>
(apply assoc_prem1; search P1) (apply pick_prem1; search P1)
|| (apply assoc_prem2; search P2) || (apply pick_prem2; search P2)
end end
in match goal with in match goal with
| [ |- ?P /\ _ --> _ ] => search P | [ |- ?P /\ _ --> _ ] => search P
...@@ -637,7 +637,7 @@ Ltac search_prem tac := ...@@ -637,7 +637,7 @@ Ltac search_prem tac :=
(** To understand how [search_prem] works, we turn first to the final [match]. If the premise begins with a conjunction, we call the [search] procedure on each of the conjuncts, or only the first conjunct, if that already yields a case where [tac] does not fail. The call [search P] expects and maintains the invariant that the premise is of the form [P /\ Q] for some [Q]. We pass [P] explicitly as a kind of decreasing induction measure, to avoid looping forever when [tac] always fails. The second [match] case calls a commutativity lemma to realize this invariant, before passing control to [search]. The final [match] case tries applying [tac] directly and then, if that fails, changes the form of the goal by adding an extraneous [True] conjunct and calls [tac] again. The %\index{tactics!progress}%[progress] tactical fails when its argument tactic succeeds without changing the current subgoal. (** To understand how [search_prem] works, we turn first to the final [match]. If the premise begins with a conjunction, we call the [search] procedure on each of the conjuncts, or only the first conjunct, if that already yields a case where [tac] does not fail. The call [search P] expects and maintains the invariant that the premise is of the form [P /\ Q] for some [Q]. We pass [P] explicitly as a kind of decreasing induction measure, to avoid looping forever when [tac] always fails. The second [match] case calls a commutativity lemma to realize this invariant, before passing control to [search]. The final [match] case tries applying [tac] directly and then, if that fails, changes the form of the goal by adding an extraneous [True] conjunct and calls [tac] again. The %\index{tactics!progress}%[progress] tactical fails when its argument tactic succeeds without changing the current subgoal.
The [search] function itself tries the same tricks as in the last case of the final [match]. Additionally, if neither works, it checks if [P] is a conjunction. If so, it calls itself recursively on each conjunct, first applying associativity lemmas to maintain the goal-form invariant. The [search] function itself tries the same tricks as in the last case of the final [match], using the [||] operator as a shorthand for trying one tactic and then, if the first fails, trying another. Additionally, if neither works, it checks if [P] is a conjunction. If so, it calls itself recursively on each conjunct, first applying associativity lemmas to maintain the goal-form invariant.
We will also want a dual function [search_conc], which does tree search through an [imp] conclusion. *) We will also want a dual function [search_conc], which does tree search through an [imp] conclusion. *)
...@@ -647,8 +647,8 @@ Ltac search_conc tac := ...@@ -647,8 +647,8 @@ Ltac search_conc tac :=
|| (apply and_True_conc; tac) || (apply and_True_conc; tac)
|| match P with || match P with
| ?P1 /\ ?P2 => | ?P1 /\ ?P2 =>
(apply assoc_conc1; search P1) (apply pick_conc1; search P1)
|| (apply assoc_conc2; search P2) || (apply pick_conc2; search P2)
end end
in match goal with in match goal with
| [ |- _ --> ?P /\ _ ] => search P | [ |- _ --> ?P /\ _ ] => search P
...@@ -718,7 +718,7 @@ Print t2. ...@@ -718,7 +718,7 @@ Print t2.
(** %\vspace{-.15in}% [[ (** %\vspace{-.15in}% [[
t2 = t2 =
fun P Q : Prop => fun P Q : Prop =>
comm_prem (assoc_prem1 (assoc_prem2 (False_prem (P:=P /\ P /\ Q) (P /\ Q)))) comm_prem (pick_prem1 (pick_prem2 (False_prem (P:=P /\ P /\ Q) (P /\ Q))))
: forall P Q : Prop, Q /\ (P /\ False) /\ P --> P /\ Q : forall P Q : Prop, Q /\ (P /\ False) /\ P --> P /\ Q
]] ]]
...@@ -750,7 +750,7 @@ fun (P : nat -> Prop) (Q : Prop) => ...@@ -750,7 +750,7 @@ fun (P : nat -> Prop) (Q : Prop) =>
and_True_prem and_True_prem
(ex_prem (P:=fun x : nat => P x /\ Q) (ex_prem (P:=fun x : nat => P x /\ Q)
(fun x : nat => (fun x : nat =>
assoc_prem2 pick_prem2
(Match (P:=Q) (Match (P:=Q)
(and_True_conc (and_True_conc
(ex_conc (fun x0 : nat => P x0) x (ex_conc (fun x0 : nat => P x0) x
......
...@@ -129,7 +129,7 @@ We have also already seen the definition of [True]. For a demonstration of a lo ...@@ -129,7 +129,7 @@ We have also already seen the definition of [True]. For a demonstration of a lo
: Prop -> Prop : Prop -> Prop
]] ]]
We see that [not] is just shorthand for implication of [False]. We can use that fact explicitly in proofs. The syntax [~ P] expands to [not P]. *) We see that [not] is just shorthand for implication of [False]. We can use that fact explicitly in proofs. The syntax [~ P] %(written with a tilde in ASCII)% expands to [not P]. *)
Theorem arith_neq' : ~ (2 + 2 = 5). Theorem arith_neq' : ~ (2 + 2 = 5).
(* begin thide *) (* begin thide *)
......
...@@ -91,7 +91,7 @@ Check Type. ...@@ -91,7 +91,7 @@ Check Type.
In the outputs of our first [Check] query, we see that the type level of [Set]'s type is [(0)+1]. Here [0] stands for the level of [Set], and we increment it to arrive at the level that _classifies_ [Set]. In the outputs of our first [Check] query, we see that the type level of [Set]'s type is [(0)+1]. Here [0] stands for the level of [Set], and we increment it to arrive at the level that _classifies_ [Set].
In the second query's output, we see that the occurrence of [Type] that we check is assigned a fresh%\index{universe variable}% _universe variable_ [Top.3]. The output type increments [Top.3] to move up a level in the universe hierarchy. As we write code that uses definitions whose types mention universe variables, unification may refine the values of those variables. Luckily, the user rarely has to worry about the details. In the third query's output, we see that the occurrence of [Type] that we check is assigned a fresh%\index{universe variable}% _universe variable_ [Top.3]. The output type increments [Top.3] to move up a level in the universe hierarchy. As we write code that uses definitions whose types mention universe variables, unification may refine the values of those variables. Luckily, the user rarely has to worry about the details.
Another crucial concept in CIC is%\index{predicativity}% _predicativity_. Consider these queries. *) Another crucial concept in CIC is%\index{predicativity}% _predicativity_. Consider these queries. *)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment