-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathTactics.v
263 lines (236 loc) · 9.84 KB
/
Tactics.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
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
Require Import Kami.Lib.EclecticLib Kami.Syntax Kami.Properties.
Ltac struct_get_field_ltac packet name :=
let val := eval cbv in (struct_get_field_index packet name) in
match val with
| Some ?x => exact (ReadStruct packet x)
| None =>
let newstr := constr:(("get field not found in struct" ++ name)%string) in
fail 0 newstr
| _ =>
let newstr := constr:(("major error - struct_get_field_index not reducing " ++ name)%string) in
fail 0 newstr
end.
Ltac struct_set_field_ltac packet name newval :=
let val := eval cbv in (struct_get_field_index packet name) in
match val with
| Some ?x => exact (UpdateStruct packet x newval)
| None =>
let newstr := constr:(("set field not found in struct " ++ name)%string) in
fail 0 newstr
| _ =>
let newstr := constr:(("major error - struct_set_field_index not reducing " ++ name)%string) in
fail 0 newstr
end.
Local Ltac constructor_simpl :=
econstructor; eauto; simpl; unfold not; intros.
Ltac destruct_string_dec :=
repeat match goal with
| H: context[string_dec ?P%string ?Q%string] |- _ =>
destruct (string_dec P Q)
| |- context[string_dec ?P%string ?Q%string] =>
destruct (string_dec P Q)
end.
Local Ltac process_append :=
repeat match goal with
| H: (_ ++ _)%string = (_ ++ _)%string |- _ =>
rewrite <- ?append_assoc in H; cbn [append] in H
| |- (_ ++ _)%string = (_ ++ _)%string =>
rewrite <- ?append_assoc; cbn [append]
end;
repeat match goal with
| H: (?a ++ ?b)%string = (?a ++ ?c)%string |- _ =>
apply append_remove_prefix in H; subst
| H: (?a ++ ?b)%string = (?c ++ ?b)%string |- _ =>
apply append_remove_suffix in H; subst
| |- (?a ++ ?b)%string = (?a ++ ?c)%string =>
apply append_remove_prefix
| |- (?a ++ ?b)%string = (?c ++ ?b)%string =>
apply append_remove_suffix
| H: (?a ++ (String ?x ?b))%string = (?c ++ (String ?y ?d))%string |- _ =>
apply (f_equal string_rev) in H;
rewrite (string_rev_append a (String x b)), (string_rev_append c (String y d)) in H;
cbn [string_rev] in H;
rewrite <- ?append_assoc in H; cbn [append] in H
end.
Local Ltac finish_append :=
auto; try (apply InSingleton || discriminate || tauto || congruence).
Ltac discharge_append :=
simpl; unfold getBool in *; process_append; finish_append.
Goal forall (a b c: string),
(a ++ "a" <> a ++ "b"
/\ a ++ "a" ++ b <> c ++ "b" ++ b
/\ a ++ "a" ++ "b" <> a ++ "a" ++ "c"
/\ "a" ++ a <> "b" ++ b
/\ (a ++ "a") ++ b <> a ++ "b" ++ a
/\ (a ++ (b ++ "b")) ++ "c" <> (a ++ b) ++ "d")%string.
Proof. intuition idtac; discharge_append. Qed.
Ltac discharge_DisjKey :=
repeat match goal with
| |- DisjKey _ _ =>
rewrite (DisjKeyWeak_same string_dec); unfold DisjKeyWeak; simpl; intros
| H: _ \/ _ |- _ => destruct H; subst
end; discharge_append.
Ltac discharge_wf :=
repeat match goal with
| |- @WfMod _ _ => constructor_simpl
| |- @WfConcat _ _ => constructor_simpl
| |- _ /\ _ => constructor_simpl
| |- @WfConcatActionT _ _ _ _ _ => constructor_simpl
| |- @WfBaseModule _ _ => constructor_simpl
| |- @WfActionT _ _ _ (convertLetExprSyntax_ActionT ?e) => apply WfLetExprSyntax
| |- @WfActionT _ _ _ _ => constructor_simpl
| |- NoDup _ => constructor_simpl
| H: _ \/ _ |- _ => destruct H; subst; simpl
| |- forall _, _ => intros
| |- _ -> _ => intros
| H: In _ (getAllMethods _) |- _ => simpl in H;inversion H;subst;clear H;simpl
end;
discharge_DisjKey.
Ltac discharge_wf_new :=
repeat match goal with
| |- @WfBaseModule_new _ _ => unfold WfBaseModule_new
| |- @WfMod_new _ _ => constructor_simpl
| |- _ /\ _ => constructor_simpl
| |- @WfActionT_new _ _ _ (convertLetExprSyntax_ActionT ?e) => apply WfLetExprSyntax
| |- @WfActionT _ _ _ (convertLetExprSyntax_ActionT ?e) => apply WfLetExprSyntax
| |- NoDup _ => constructor_simpl
| H: _ \/ _ |- _ => destruct H; subst; simpl
| |- forall _, _ => intros
| |- _ -> _ => intros
| H: In _ (getAllMethods _) |- _ => simpl in H;inversion H;subst;clear H;simpl
| |- _ => unfold lookup; simpl; repeat rewrite strip_pref
end;
discharge_DisjKey.
Lemma string_dec_refl {A} : forall (s: string) (T E: A),
(if String.eqb s s then T else E) = T.
Proof.
intros; rewrite String.eqb_refl; auto.
Qed.
Lemma string_dec_neq {A} : forall (s1 s2: string) (T E: A),
s1 <> s2 ->
(if String.eqb s1 s2 then T else E) = E.
Proof.
intros.
rewrite <- String.eqb_neq in H; rewrite H; auto.
Qed.
Ltac discharge_string_dec :=
repeat (rewrite string_dec_refl || rewrite string_dec_neq by (intros ?; discharge_append)).
Ltac discharge_NoSelfCall :=
unfold NoSelfCallBaseModule, NoSelfCallRulesBaseModule, NoSelfCallMethsBaseModule; split; auto; simpl; intros;
repeat match goal with
| H: _ \/ _ |- _ => destruct H; subst
| H: False |- _ => exfalso; apply H
| |- NoCallActionT _ (convertLetExprSyntax_ActionT _) => apply LetExprNoCallActionT
| _ => constructor; auto; simpl; try intro; discharge_DisjKey
end.
Ltac unfold_beta_head a :=
let new :=
lazymatch a with
| ?h _ _ _ _ _ _ _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ _ _ _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ _ _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ _ => eval cbv beta delta [h] in a
| ?h _ _ => eval cbv beta delta [h] in a
| ?h _ => eval cbv beta delta [h] in a
end in
exact new.
Ltac discharge_SemAction :=
match goal with
| |- SemAction _ _ _ _ ?meths _ =>
repeat match goal with
| |- SemAction ?o ?act ?reads ?news ?calls ?retv =>
let act' := constr:(ltac:(unfold_beta_head act)) in
change (SemAction o act' reads news calls retv)
| |- SemAction _ (@IfElse _ _ ?p _ _ _ _) _ _ _ _ => eapply SemAction_if_split
| |- if ?P then SemAction _ _ _ _ _ _ else SemAction _ _ _ _ _ _ =>
case_eq P;
let H := fresh in
intros H; rewrite ?H in *; cbn [evalExpr] in *; try discriminate
| |- SemAction _ (convertLetExprSyntax_ActionT _) _ _ _ _ => eapply convertLetExprSyntax_ActionT_same
| |- SemAction _ _ _ _ _ _ => econstructor
end;
rewrite ?key_not_In_fst; unfold not; intros; cbn [evalExpr evalConstT] in *;
repeat match goal with
| |- In _ _ => simpl; auto
| |- ?a = ?a => reflexivity
| |- meths = _ => eauto
end;
simpl in *; try (discriminate || congruence); eauto; simpl in *; discharge_DisjKey
end.
Ltac simplify_simulatingRule name :=
right;
exists name;
eexists; split; [eauto| do 2 eexists; split; [discharge_SemAction|]].
Ltac simplify_nilStep :=
left; split; auto; simpl in *;
discharge_string_dec.
Local Ltac discharge_init :=
repeat econstructor;
try match goal with
| |- match ?P in _ = Y return _ with
| eq_refl => _
end = _ => is_evar P; match type of P with
| ?tp = _ => unify P (@eq_refl _ tp)
end
end; simpl; eauto.
Ltac clean_hyp_step :=
match goal with
| |- NoSelfCallBaseModule _ => discharge_NoSelfCall
| H: DisjKey _ _ |- _ => clear H
| H: key_not_In _ _ |- _ => clear H
| H: ?a = ?a |- _ => clear H
| H: False |- _ => exfalso; apply H
| H: ?a <> ?a |- _ => exfalso; apply (H eq_refl)
| H: _ \/ _ |- _ => destruct H; subst
| H: _ /\ _ |- _ => destruct H; subst
| H: exists x, _ |- _ => let y := fresh x in destruct H as [y ?]
| H: (?A, ?B) = (?P, ?Q) |- _ =>
apply inversionPair in H; destruct H as [? ?]; subst
| H: existT ?a ?b ?c1 = existT ?a ?b ?c2 |- _ => apply Eqdep.EqdepTheory.inj_pair2 in H; subst
| H: existT ?a ?b1 ?c1 = existT ?a ?b2 ?c2 |- _ => apply inversionExistT in H;
destruct H as [? ?]; subst
| H: (?a ++ ?b)%string = (?a ++ ?c)%string |- _ =>
apply append_remove_prefix in H; subst
| H: ?a = ?b |- _ => discriminate
| H: SemAction _ (convertLetExprSyntax_ActionT ?e) _ _ _ _ |- _ =>
apply convertLetExprSyntax_ActionT_full in H
| H: SemAction _ _ _ _ _ _ |- _ =>
apply inversionSemAction in H
| H: if ?P then _ else _ |- _ => case_eq P; let i := fresh in intros i; rewrite ?i in *
| H: Forall2 _ _ _ |- _ => inv H; dest
| H: RegT |- _ => destruct H as [? [? ?]];
repeat (unfold fst, snd, projT1, projT2 in *; subst)
| H: In _ _ |- _ => simpl in H
| |- exists rspec : list RegT,
Forall2 _ _ _ /\ _ _ _ => discharge_init
end.
Ltac clean_hyp := simpl in *; repeat clean_hyp_step.
Ltac discharge_CommonRegister disjReg :=
match goal with
| |- exists k: string, _ /\ _ =>
exists disjReg; simpl; auto; tauto
| _ => idtac
end.
Ltac discharge_CommonRegisterAuto :=
match goal with
| |- exists k: string, _ /\ _ =>
eexists; simpl; eauto; tauto
| _ => idtac
end.
Ltac discharge_simulationWf mySimRel :=
apply simulationGeneral with (simRel := mySimRel); auto; simpl; intros;
try match goal with
| H: mySimRel _ _ |- _ => inv H
end;
clean_hyp; auto; clean_hyp.
Ltac discharge_simulation mySimRel :=
apply simulation with (simRel := mySimRel); auto; simpl; intros;
try match goal with
| |- WfBaseModule _ _ => discharge_wf
| H: mySimRel _ _ |- _ => inv H
end;
clean_hyp; auto; clean_hyp.