forked from mit-plv/koika
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathEqDec.v
76 lines (65 loc) · 2.26 KB
/
EqDec.v
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
(*! Utilities | Decidable equality typeclass !*)
Require Import Coq.Strings.String.
Require Import Coq.Vectors.Vector.
Class EqDec (T: Type) :=
{ eq_dec: forall t1 t2: T, { t1 = t2 } + { t1 <> t2 } }.
Definition beq_dec {A} {EQ: EqDec A} a1 a2 : bool :=
if eq_dec a1 a2 then true else false.
Lemma beq_dec_iff {A} (EQ: EqDec A) a1 a2 :
beq_dec a1 a2 = true <-> a1 = a2.
Proof.
unfold beq_dec; destruct eq_dec; subst.
- firstorder.
- split; intro; (eauto || congruence).
Qed.
Lemma beq_dec_false_iff {A} (EQ: EqDec A) a1 a2 :
beq_dec a1 a2 = false <-> a1 <> a2.
Proof.
unfold beq_dec; destruct eq_dec; subst;
intuition congruence.
Qed.
Hint Extern 1 (EqDec _) => econstructor; decide equality : typeclass_instances.
Hint Extern 1 ({ _ = _ } + { _ <> _ }) => apply eq_dec : typeclass_instances.
Instance EqDec_bool : EqDec bool := _.
Instance EqDec_ascii : EqDec Ascii.ascii := _.
Instance EqDec_string : EqDec string := _.
Instance EqDec_unit : EqDec unit := _.
Instance EqDec_nat : EqDec nat := {| eq_dec := PeanoNat.Nat.eq_dec |}.
Instance EqDec_pair A B `{EqDec A} `{EqDec B} : EqDec (A * B) := _.
Instance EqDec_option A `{EqDec A} : EqDec (option A) := _.
Instance EqDec_vector A (sz: nat) {EQ: EqDec A}: EqDec (Vector.t A sz).
Proof. econstructor; intros; eapply Vector.eq_dec; apply beq_dec_iff. Defined.
Instance EqDec_eq_true {A} (f: A -> bool) (a: A) : EqDec (f a = true).
Proof. constructor; left; apply Eqdep_dec.UIP_dec, eq_dec. Qed.
Import EqNotations.
Lemma eq_dec_rew_type_family {T} {EQ: EqDec T} (family: T -> Type):
forall (t: T) (Heq: t = t) (a: family t),
rew Heq in a = a.
Proof.
intros. apply eq_sym, Eqdep_dec.eq_rect_eq_dec, eq_dec.
Qed.
Lemma eq_rect_eqdec_irrel {A} {EQ: EqDec A} (x: A) {P: A -> Type} {px: P x} {y: A}:
forall (pr2 pr1: x = y),
eq_rect x P px y pr1 =
eq_rect x P px y pr2.
Proof.
destruct pr1; cbn; intros.
symmetry; apply eq_dec_rew_type_family.
Qed.
Lemma eq_dec_refl {A} {EQ: EqDec A}:
forall a, eq_dec a a = left eq_refl.
Proof.
intros.
destruct eq_dec.
- generalize e; apply Eqdep_dec.K_dec_type.
+ apply eq_dec.
+ reflexivity.
- congruence.
Qed.
Lemma beq_dec_refl {A} {EQ: EqDec A}:
forall a, beq_dec a a = true.
Proof.
intros.
unfold beq_dec.
rewrite eq_dec_refl; reflexivity.
Qed.