forked from vafeiadis/hahn
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHahnSorted.v
179 lines (154 loc) · 5.43 KB
/
HahnSorted.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
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
(******************************************************************************)
(** * Lemmas about sorted lists *)
(******************************************************************************)
Require Import HahnBase HahnSets HahnList HahnRelationsBasic HahnEquational.
Require Export Sorted Setoid.
Set Implicit Arguments.
Add Parametric Morphism A : (@StronglySorted A) with signature
inclusion ==> eq ==> Basics.impl as StronglySorted_mori.
Proof.
unfold inclusion; red; ins; desf.
induction H0; econs; ins.
rewrite Forall_forall in *; ins; desf; eauto.
Qed.
Add Parametric Morphism A : (@StronglySorted A) with signature
same_relation ==> eq ==> iff as StronglySorted_more.
Proof.
by split; [rewrite (proj1 H)|rewrite (proj2 H)].
Qed.
Lemma StronglySorted_cons_iff A (r : relation A) (a : A) (l : list A) :
StronglySorted r (a :: l) <-> StronglySorted r l /\ Forall (r a) l.
Proof.
split; ins; desf; auto using SSorted_cons, StronglySorted_inv.
Qed.
Lemma StronglySorted_app_iff A (r: relation A) l1 l2 :
StronglySorted r (l1 ++ l2) <->
StronglySorted r l1 /\ StronglySorted r l2 /\
(forall x1 (IN1: In x1 l1) x2 (IN2: In x2 l2), r x1 x2).
Proof.
induction l1; ins; [by intuition; vauto|].
rewrite !StronglySorted_cons_iff, IHl1, Forall_app.
clear; intuition; desf; eauto.
all: rewrite Forall_forall in *; eauto.
Qed.
Lemma StronglySorted_app A (r: relation A) l1 l2
(S1: StronglySorted r l1) (S2: StronglySorted r l2)
(L: forall x1 (IN1: In x1 l1) x2 (IN2: In x2 l2), r x1 x2):
StronglySorted r (l1 ++ l2).
Proof.
by apply StronglySorted_app_iff.
Qed.
Lemma StronglySorted_app_l A r (l l': list A):
StronglySorted r (l ++ l') -> StronglySorted r l.
Proof.
rewrite StronglySorted_app_iff; tauto.
Qed.
Lemma StronglySorted_app_r A r (l l': list A):
StronglySorted r (l ++ l') -> StronglySorted r l'.
Proof.
rewrite StronglySorted_app_iff; tauto.
Qed.
Lemma StronglySorted_filter A r f (l : list A):
StronglySorted r l -> StronglySorted r (filter f l).
Proof.
induction l; ins; eauto; desf; rewrite StronglySorted_cons_iff in *; desf.
all: splits; desf; eauto using Forall_filter.
Qed.
Lemma StronglySorted_filterP A r f (l : list A):
StronglySorted r l -> StronglySorted r (filterP f l).
Proof.
induction l; ins; eauto; desf; rewrite StronglySorted_cons_iff in *; desf.
all: splits; desf; eauto using Forall_filterP.
Qed.
Lemma NoDup_StronglySorted A (r: relation A) (IRR: irreflexive r)
l (SS: StronglySorted r l):
NoDup l.
Proof.
induction l; ins.
apply StronglySorted_inv in SS; desc.
rewrite Forall_forall in *.
constructor; eauto.
Qed.
Hint Resolve NoDup_StronglySorted : hahn.
Hint Resolve SSorted_nil : hahn.
Hint Resolve StronglySorted_filterP : hahn.
Hint Resolve StronglySorted_filter : hahn.
Lemma sorted_perm_eq : forall A (cmp: A -> A -> Prop)
(TRANS: transitive cmp)
(ANTIS: antisymmetric cmp)
l l' (P: Permutation l l')
(S : StronglySorted cmp l) (S' : StronglySorted cmp l'), l = l'.
Proof.
induction l; ins.
by apply Permutation_nil in P; desf.
assert (X: In a l') by eauto using Permutation_in, Permutation_sym, in_eq.
apply In_split2 in X; desf; apply Permutation_cons_app_inv in P.
destruct l1; ins; [by inv S; inv S'; eauto using f_equal|].
assert (X: In a0 l) by eauto using Permutation_in, Permutation_sym, in_eq.
inv S; inv S'; rewrite Forall_forall in *; ins.
destruct X0; left; apply ANTIS; eauto with hahn.
Qed.
Lemma StronglySorted_eq A (r: relation A) (ORD: strict_partial_order r) l l'
(SS: StronglySorted r l) (SS': StronglySorted r l')
(EQ: forall x, In x l <-> In x l'): l = l'.
Proof.
red in ORD; desc.
eapply sorted_perm_eq; eauto using NoDup_Permutation with hahn.
Qed.
Lemma StronglySorted_restr A cond (r: relation A) l:
StronglySorted (restr_rel cond r) l -> StronglySorted r l.
Proof.
induction l; ins; vauto.
rewrite StronglySorted_cons_iff, ForallE in *.
firstorder.
Qed.
Lemma restr_StronglySorted A cond (r: relation A) l:
StronglySorted r l ->
Forall cond l ->
StronglySorted (restr_rel cond r) l.
Proof.
induction l; ins.
rewrite StronglySorted_cons_iff, ForallE in *.
firstorder.
Qed.
Fixpoint isort A (r : relation A) l :=
match l with
| nil => nil
| x :: l =>
let l' := isort r l in
filterP (fun y => r y x) l' ++ x :: filterP (fun y => ~ r y x) l'
end.
Hint Resolve SSorted_nil : hahn.
Hint Resolve StronglySorted_filterP : hahn.
Lemma in_isort_iff A x (r : relation A) l : In x (isort r l) <-> In x l.
Proof.
induction l; ins; rewrite in_app_iff; ins; in_simp; rewrite IHl.
tauto.
Qed.
Lemma StronglySorted_isort A (r : relation A)
(TOT: strict_total_order (fun _ : A => True) r)
l (ND: NoDup l) :
StronglySorted r (isort r l).
Proof.
destruct TOT as [[IRR T] TOT].
induction l; ins; vauto.
rewrite nodup_cons in *; desc.
apply StronglySorted_app; ins; in_simp; eauto with hahn.
econs; eauto with hahn.
apply Forall_forall; ins; in_simp.
all: rewrite in_isort_iff in *; desf.
forward eapply TOT with (a:=x) (b:=a); try red; ins; desf; eauto.
forward eapply TOT with (a:=x1) (b:=x2); try red; ins; desf; eauto.
exfalso; eauto.
Qed.
Lemma isort_Permutation A (r : relation A) l : Permutation l (isort r l).
Proof.
induction l; ins.
apply Permutation_cons_app.
now rewrite <- Permutation_filterP_split.
Qed.
Lemma isort_NoDup A (r : relation A) l : NoDup l <-> NoDup (isort r l).
Proof.
split; apply Permutation_nodup;
eauto using Permutation_sym, isort_Permutation.
Qed.