-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathML_SP_SoundDomain.v
923 lines (865 loc) · 28.6 KB
/
ML_SP_SoundDomain.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
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
(***************************************************************************
* Preservation and Progress for mini-ML with structural polymorphism *
* Jacques Garrigue, May 2008 *
***************************************************************************)
Set Implicit Arguments.
Require Import Lib_FinSet Lib_FinSetImpl Metatheory List ListSet Arith.
Require Import ML_SP_Soundness.
Require Import Omega.
Section ListSet.
Variable A : Type.
Hypothesis eq_dec : forall x y:A, {x = y} + {x <> y}.
Definition set_incl : forall (s1 s2 : list A),
sumbool (incl s1 s2) (~incl s1 s2).
intros.
induction s1. left*.
case_eq (set_mem eq_dec a s2); intros.
destruct IHs1.
left*; intros x Hx. simpl in Hx; destruct Hx.
subst; apply* set_mem_correct1. apply* i.
right*.
right*. intro; elim (set_mem_complete1 eq_dec _ _ H).
apply* H0.
Defined.
End ListSet.
Module Cstr.
Definition attr := nat.
Definition eq_dec := eq_nat_dec.
Inductive ksort := Ksum | Kprod | Kbot.
Record cstr_impl : Set := C {
cstr_sort : ksort;
cstr_low : list nat;
cstr_high : option (list nat) }.
Definition cstr := cstr_impl.
Definition valid c :=
cstr_sort c <> Kbot /\
match cstr_high c with
| None => True
| Some h => incl (cstr_low c) h
end.
Definition le_sort s1 s2 :=
s1 = Kbot \/ s1 = s2.
Definition entails c1 c2 :=
le_sort (cstr_sort c1) (cstr_sort c2) /\
incl (cstr_low c2) (cstr_low c1) /\
match cstr_high c2 with
| None => True
| Some h2 =>
match cstr_high c1 with
| None => False
| Some h1 => incl h1 h2
end
end.
Lemma entails_refl : forall c, entails c c.
Proof.
intros.
split. right*.
split. apply incl_refl.
destruct* (cstr_high c).
Qed.
Lemma entails_trans : forall c1 c2 c3,
entails c1 c2 -> entails c2 c3 -> entails c1 c3.
Proof.
introv. intros [CS1 [CL1 CH1]] [CS2 [CL2 CH2]].
split.
unfold le_sort.
destruct* CS1.
rewrite H. destruct* CS2.
split. apply* incl_tran.
clear CL1 CL2.
destruct* (cstr_high c3).
destruct* (cstr_high c2).
destruct* (cstr_high c1).
Qed.
Definition unique c v := set_mem eq_nat_dec v (cstr_low c).
Definition sort_lub s1 s2 :=
match s1, s2 with
| Ksum, Ksum => Ksum
| Kprod, Kprod => Kprod
| _, _ => Kbot
end.
Definition lub c1 c2 :=
C (sort_lub (cstr_sort c1) (cstr_sort c2))
(set_union eq_nat_dec (cstr_low c1) (cstr_low c2))
(match cstr_high c1, cstr_high c2 with
| None, h => h
| h, None => h
| Some s1, Some s2 => Some (set_inter eq_nat_dec s1 s2)
end).
Lemma ksort_dec : forall s, {s=Kbot} + {s<>Kbot}.
intro; destruct* s; right*; intro; discriminate.
Qed.
Hint Resolve incl_tran : core.
Lemma entails_lub : forall c1 c2 c,
(entails c c1 /\ entails c c2) <-> entails c (lub c1 c2).
Proof.
unfold entails, lub; simpl; split; intros.
intuition.
destruct (ksort_dec (cstr_sort c)).
rewrite e; left*.
destruct* H.
destruct* H0.
rewrite <- H, <- H0.
right*; destruct* (cstr_sort c).
intros x Hx; destruct* (set_union_elim _ _ _ _ Hx).
case_rewrite R1 (cstr_high c1);
case_rewrite R2 (cstr_high c2);
try case_rewrite R3 (cstr_high c); intuition.
intros x Hx; apply set_inter_intro. apply* H4. apply* H5.
destruct H as [HS [HL HH]].
split; split; try split;
try (intros x Hx; apply HL;
solve [ apply* set_union_intro2 | apply* set_union_intro1 ]);
try solve [
case_rewrite R1 (cstr_high c1);
case_rewrite R2 (cstr_high c2);
try case_rewrite R3 (cstr_high c); intuition;
intros x Hx; destruct* (set_inter_elim _ _ _ _ (HH _ Hx))];
clear -HS; destruct HS; try solve [left*];
rewrite H;
unfold le_sort; destruct (cstr_sort c1); destruct* (cstr_sort c2).
Qed.
Definition valid_dec : forall c, sumbool (valid c) (~valid c).
intros.
unfold valid.
destruct* (ksort_dec (cstr_sort c)).
case_eq (cstr_high c); intros.
destruct (set_incl eq_nat_dec (cstr_low c) l). left*.
right*.
auto.
Defined.
Lemma entails_unique : forall c1 c2 v,
entails c1 c2 -> unique c2 v = true -> unique c1 v = true.
Proof.
unfold entails, unique.
intros.
destruct H as [_ [H _]].
apply set_mem_correct2.
apply H.
apply* set_mem_correct1.
Qed.
Lemma entails_valid : forall c1 c2,
entails c1 c2 -> valid c1 -> valid c2.
Proof.
unfold entails, valid.
intros.
destruct H as [HS [HL HH]]; destruct H0.
destruct* HS.
rewrite <- H1. split2*.
case_rewrite R1 (cstr_high c1); case_rewrite R2 (cstr_high c2);
auto*.
Qed.
End Cstr.
Section NoDup.
Fixpoint nodup (l:list nat) :=
match l with
| nil => nil
| v :: l' =>
if In_dec eq_nat_dec v l' then nodup l' else v :: nodup l'
end.
Lemma nodup_elts : forall a l, In a l <-> In a (nodup l).
Proof.
induction l; split2*; simpl; intro; destruct IHl.
destruct H. subst.
destruct* (In_dec eq_nat_dec a l).
destruct* (In_dec eq_nat_dec a0 l).
destruct* (In_dec eq_nat_dec a0 l).
simpl in H; destruct* H.
Qed.
Lemma NoDup_nodup : forall l, NoDup (nodup l).
Proof.
induction l; simpl. constructor.
destruct* (In_dec eq_nat_dec a l).
constructor; auto.
intro Ha. elim n. apply (proj2 (nodup_elts _ _) Ha).
Qed.
End NoDup.
Module Const.
Inductive ops : Set :=
| tag : Cstr.attr -> ops
| matches : forall (l:list Cstr.attr), NoDup l -> ops
| record : forall (l:list Cstr.attr), NoDup l -> ops
| sub : Cstr.attr -> ops
| recf : ops.
Definition const := ops.
Definition arity op :=
match op with
| tag _ => 1
| @matches l _ => length l
| @record l _ => length l
| sub _ => 0
| recf => 1
end.
End Const.
Module Sound := MkSound(Cstr)(Const).
Import Sound.
Import Infra.
Import Defs.
Lemma list_snd_combine : forall (A B:Set) (l1:list A) (l2:list B),
length l1 = length l2 ->
list_snd (combine l1 l2) = l2.
Proof.
induction l1; destruct l2; intros; try discriminate. reflexivity.
inversions H.
simpl. rewrite* (IHl1 l2).
Qed.
Module Delta.
Definition matches_arg n := typ_arrow (typ_bvar n) (typ_bvar 1).
Lemma valid_tag : forall s t,
s <> Cstr.Kbot -> Cstr.valid (Cstr.C s (t::nil) None).
Proof.
intros. split2*. compute. auto.
Qed.
Lemma ksum : Cstr.Ksum <> Cstr.Kbot.
intro; discriminate.
Qed.
Lemma kprod : Cstr.Kprod <> Cstr.Kbot.
intro; discriminate.
Qed.
Lemma coherent_tag : forall s t,
coherent (Cstr.C s (t::nil) None) ((t, typ_bvar 0) :: nil).
Proof.
intros s t x; intros.
destruct H0; destruct H1; try contradiction.
inversions H0. inversions* H1.
Qed.
Lemma valid_matches : forall s l,
s <> Cstr.Kbot -> Cstr.valid (Cstr.C s nil (Some l)).
Proof.
intros; split2*. simpl*.
Qed.
Lemma coherent_matches : forall s n l,
NoDup l ->
coherent (Cstr.C s nil (Some l))
(combine l (map typ_bvar (seq n (length l)))).
Proof.
intros; intro; intros.
clear H0; revert H1 H2; gen n.
induction H; simpl; intros. elim H1.
destruct H1. inversions H1.
destruct H2. inversions* H2.
elim (H (in_combine_l _ _ _ _ H2)).
destruct H2. inversions H2.
elim (H (in_combine_l _ _ _ _ H1)).
apply* (IHNoDup (S n)).
Qed.
Definition type (op:Const.const) :=
match op with
| Const.tag t =>
Sch (typ_arrow (typ_bvar 0) (typ_bvar 1))
(None :: Some (Kind (valid_tag t ksum) (@coherent_tag _ t)) :: nil)
| @Const.matches l ND =>
Sch (fold_right typ_arrow (typ_arrow (typ_bvar 0) (typ_bvar 1))
(map matches_arg (seq 2 (length l))))
(Some (Kind (@valid_matches _ l ksum) (coherent_matches 2 ND)) ::
map (fun _ => None) (seq 0 (S (length l))))
| @Const.record l ND =>
Sch (fold_right typ_arrow (typ_bvar 0) (map typ_bvar (seq 1 (length l))))
(Some (Kind (@valid_matches _ l kprod) (coherent_matches 1 ND)) ::
map (fun _ => None) (seq 0 (length l)))
| Const.sub f =>
Sch (typ_arrow (typ_bvar 1) (typ_bvar 0))
(None :: Some (Kind (valid_tag f kprod) (@coherent_tag _ f)) :: nil)
| Const.recf =>
let T := typ_arrow (typ_bvar 0) (typ_bvar 1) in
Sch (typ_arrow (typ_arrow T T) T) (None :: None :: nil)
end.
Definition trm_default := trm_abs trm_def.
Fixpoint record_args (t : trm) tl {struct t} : list nat * list trm :=
match t with
| trm_app t1 t2 => record_args t1 (t2 :: tl)
| trm_cst (@Const.record l _) => (l, tl)
| _ => (nil, nil)
end.
Lemma record_args_ok : forall l nd tl tl',
record_args (const_app (@Const.record l nd) tl) tl' = (l, tl++tl').
Proof.
induction tl using rev_ind; simpl; intros. auto.
rewrite const_app_app, app_ass; simpl.
apply* IHtl.
Qed.
Definition is_record c :=
match c with
| Const.record _ => true
| _ => false
end.
Lemma record_args_other : forall c tl tl',
is_record c = false -> record_args (const_app c tl) tl' = (nil, nil).
Proof.
induction tl using rev_ind; simpl; intros. destruct* c. discriminate.
rewrite const_app_app. simpl.
apply* IHtl.
Qed.
Definition reduce c tl (vl:list_for_n value (S(Const.arity c)) tl) :=
match c with
| Const.tag _ => trm_default
| @Const.matches l nd =>
match nth (length l) tl trm_def with
| trm_app (trm_cst (Const.tag t)) t1 =>
match index eq_nat_dec 0 t l with
| Some i => trm_app (nth i tl trm_def) t1
| None => trm_default
end
| _ => trm_default
end
| Const.record _ => trm_default
| Const.sub f =>
match tl with
| nil => trm_default
| t :: _ =>
let (l, fl) := record_args t nil in
match index eq_nat_dec 0 f l with
| Some i => nth i fl trm_default
| None => trm_default
end
end
| Const.recf =>
match tl with
| f :: a :: _ => trm_app (trm_app f (trm_app (trm_cst Const.recf) f)) a
| _ => trm_default
end
end.
Lemma term_default : term trm_default.
Proof.
unfold trm_default, trm_def.
apply (@term_abs {}).
intros. unfold trm_open; simpl. auto.
Qed.
Hint Resolve term_default : core.
Lemma value_term : forall e,
value e -> term e.
Proof.
intros. destruct H. induction H; auto.
Qed.
Hint Resolve value_term : core.
Lemma term : forall c tl vl,
term (@reduce c tl vl).
Proof.
intros.
case vl; clear vl; intros.
destruct c; simpl in *; auto.
(* matches *)
assert (length l0 < S (length l0)) by auto.
rewrite e in H.
case_eq (nth (length l0) tl trm_def); intros; auto.
destruct* t.
destruct* c.
case_eq (index eq_nat_dec 0 a l0); intros; auto.
apply term_app.
destruct (index_ok _ 0 _ _ H1).
apply value_term.
apply* (list_forall_out l).
apply* nth_In.
unfold Cstr.attr in *; omega.
assert (term (nth (length l0) tl trm_def)).
apply value_term.
apply* (list_forall_out l).
apply* nth_In.
rewrite H0 in H2.
inversion* H2.
(* sub *)
destruct* tl.
case_eq (record_args t nil); intros.
assert (list_forall value l1).
inversions l.
clear -H H3.
assert (list_forall value nil) by auto.
revert H H0. generalize (@nil trm).
destruct H3.
induction H; simpl; intros; try solve [inversions* H0].
destruct c; inversions* H.
apply* (IHvalu1 _ H1).
constructor; auto.
exists* n2.
clear -H0.
destruct* (index eq_nat_dec 0 a l0).
gen n; induction H0; simpl; intros; auto; destruct* n.
(* recf *)
inversions* l; clear l.
inversions* H; clear H.
Qed.
Lemma closed : forall c, sch_fv (Delta.type c) = {}.
Proof.
intros.
induction c; unfold sch_fv; simpl.
(* tag *)
unfold kind_fv; simpl.
repeat rewrite union_empty_l. auto.
(* matches *)
set (x := 1). unfold x at 1.
unfold kind_fv_list, kind_fv; simpl.
generalize x; clear.
induction l; simpl in *; intros;
repeat rewrite union_empty_l.
auto.
use (IHl (S x)); clear IHl. rewrite union_empty_l in H. auto.
(* record *)
set (x := 0). unfold x at 1.
unfold kind_fv_list, kind_fv; simpl.
generalize x; clear.
induction l; simpl in *; intros;
repeat rewrite union_empty_l.
auto.
use (IHl (S x)).
(* sub *)
unfold kind_fv; simpl.
repeat rewrite union_empty_l. auto.
(* recf *)
simpl.
repeat rewrite union_empty_l. auto.
Qed.
Lemma type_nth_typ_vars : forall n Xs,
n < length Xs -> Defs.type (nth n (typ_fvars Xs) typ_def).
Proof.
induction n; destruct Xs; simpl; intros; try (elimtype False; omega).
auto.
apply IHn; omega.
Qed.
Hint Extern 1 (Defs.type (nth _ (typ_fvars _) typ_def)) =>
solve [apply type_nth_typ_vars; omega] : core.
Lemma scheme : forall c, scheme (Delta.type c).
Proof.
intros. intro; intros.
unfold typ_open_vars.
destruct c; simpl in *.
(* tag *)
do 3 (destruct Xs; try discriminate).
simpl.
split2*.
unfold All_kind_types.
repeat (constructor; simpl*).
(* matches *)
rewrite map_length, seq_length in H.
split.
do 2 (destruct Xs; try discriminate).
inversion H; clear H.
replace Xs with (nil ++ Xs) by simpl*.
set (Xs0 := nil(A:=var)).
replace 2 with (S (S (length Xs0))) by simpl*.
gen Xs; generalize Xs0; induction n; simpl; intros. auto.
destruct Xs; try discriminate.
constructor.
unfold typ_fvars.
rewrite map_app.
rewrite app_nth2 by (rewrite map_length; omega).
rewrite map_length.
replace (length Xs1 - length Xs1) with 0 by omega.
simpl*.
replace (v1 :: Xs) with ((v1 :: nil) ++ Xs) by simpl*.
rewrite <- app_ass.
simpl in IHn.
replace (S (length Xs1)) with (length (Xs1 ++ v1 :: nil))
by (rewrite app_length; simpl; omega).
apply* IHn.
unfold All_kind_types.
repeat (constructor; auto).
induction (seq 1 (length l)); simpl; try constructor; simpl*.
simpl.
assert (length Xs >= 2 + length l) by omega.
clear H; revert H0.
generalize 2; induction n; simpl; intros. auto.
constructor. apply* IHn. omega.
simpl*.
(* record *)
rewrite map_length, seq_length in H.
assert (1 + length l = length Xs) by omega. clear H.
split.
assert (0 < 1) by omega.
revert H0 H.
generalize 1 Xs. clear.
induction (length l); simpl; intros. auto.
constructor. auto.
apply* IHn.
repeat (constructor; auto).
clear; generalize 0; induction (length l); simpl*.
unfold All_kind_types. simpl.
rewrite list_snd_combine by (rewrite map_length, seq_length; auto).
revert H0; generalize 1.
unfold Cstr.attr in *.
clear; induction (length l); simpl; intros. auto.
constructor. apply* IHn.
simpl*.
(* sub *)
split. simpl*.
repeat (constructor; auto).
apply type_nth_typ_vars; omega.
unfold All_kind_types. simpl.
repeat (constructor; simpl*).
Qed.
End Delta.
Module Sound2 := Sound.Mk2(Delta).
Import Sound2.
Import JudgInfra.
Import Judge.
Module SndHyp.
Lemma fold_arrow_eq : forall T1 TL1 T2 TL2 Us,
typ_open (fold_right typ_arrow T1 TL1) Us = fold_right typ_arrow T2 TL2 ->
length TL1 = length TL2 ->
typ_open T1 Us = T2 /\
list_forall2 (fun U1 U2 => typ_open U1 Us = U2) TL1 TL2.
Proof.
induction TL1; intros; destruct* TL2; simpl in *; try discriminate; auto.
inversions H.
destruct* (IHTL1 T2 TL2 Us).
Qed.
Lemma gc_lower_false : forall gc, gc_lower (false, gc) = (false, gc).
Proof. destruct gc; simpl*. Qed.
Lemma fold_app_inv : forall K E gc t tl T,
K ; E |(false,gc)|= fold_left trm_app tl t ~: T ->
exists TL,
K ; E |(false,gc)|= t ~: fold_right typ_arrow T TL /\
list_forall2 (typing (false,gc) K E) tl TL.
Proof.
induction tl using rev_ind; simpl; intros.
exists* (nil(A:=typ)).
rewrite fold_left_app in *; simpl in *.
inversions H; try discriminate. rewrite gc_lower_false in *.
destruct (IHtl (typ_arrow S T) H4).
exists (x0 ++ S :: nil).
rewrite fold_right_app; simpl.
split2*.
Qed.
Lemma map_nth : forall (A B:Set) d1 d2 (f:A->B) k l,
k < length l -> nth k (map f l) d1 = f (nth k l d2).
Proof.
induction k; intros; destruct l; simpl in *; auto; try elim (lt_n_O _ H).
apply IHk.
apply* lt_S_n.
Qed.
Hint Rewrite combine_length combine_nth : list.
Definition delta_typed_cst c :=
forall tl vl K E gc T TL,
K; E | (false, gc) |= trm_cst c ~: fold_right typ_arrow T TL ->
list_forall2 (typing (false, gc) K E) tl TL ->
K ; E |(false,gc)|= @Delta.reduce c tl vl ~: T.
Lemma delta_typed_tag : forall v, delta_typed_cst (Const.tag v).
Proof.
intros; intro; introv TypC TypA.
elimtype False.
destruct vl as [Hv _].
inversions TypC; try discriminate. clear TypC H4.
destruct* tl; try destruct* tl; try destruct* tl; try discriminate.
inversions TypA; clear TypA.
inversions H6; clear H6.
inversions H8; clear H8.
unfold sch_open in H0.
simpl in H0.
inversions H0. clear H0.
destruct H5 as [_ WK]. simpl in WK.
inversions WK; clear WK.
inversions H7; clear H7.
inversions H5; clear H5.
discriminate.
Qed.
Lemma typing_value_inv : forall gc K E t T n,
K;E |(false,gc)|= t ~: T -> valu n t ->
(exists t1, t = trm_abs t1) \/
(exists c, exists vl, t = const_app c vl /\
n <= Const.arity c /\ list_for_n value (Const.arity c - n) vl).
Proof.
intros.
gen K E T. induction H0; intros.
left*.
right*; exists c; exists (@nil trm).
split2*.
rewrite <- minus_n_n. split2*. split2*.
clear IHvalu2.
inversions H; clear H; try discriminate.
rewrite gc_lower_false in *.
destruct (IHvalu1 _ _ _ H4); clear IHvalu1.
destruct H. subst. inversion H0_.
destruct H as [c [vl [HE [HC HV]]]].
subst.
right*; exists c; exists (vl ++ t2 :: nil).
rewrite const_app_app. split2*.
split. omega.
destruct HV. split2*.
assert (value t2) by exists* n2.
apply* list_forall_app.
Qed.
Lemma value_fvar_is_const : forall K E gc t x k,
K; E |(false,gc)|= t ~: typ_fvar x ->
binds x (Some k) K ->
value t ->
match Cstr.cstr_sort (kind_cstr k) with
| Cstr.Ksum =>
exists v, exists t2, t = trm_app (trm_cst (Const.tag v)) t2
| Cstr.Kprod =>
exists l, exists nd, exists tl,
t = const_app (@Const.record l nd) tl /\ length tl = length l
| Cstr.Kbot => True
end.
Proof.
introv Typ B HV.
destruct HV as [i vi].
destruct (typing_value_inv Typ vi).
destruct H. subst. inversions Typ. discriminate.
destruct H as [c [vl [HE [HA Hvl]]]].
subst.
destruct (fold_app_inv _ _ Typ) as [TL [Typ0 TypA]]; clear Typ.
inversions Typ0; clear Typ0; try discriminate.
unfold sch_open in H0. destruct H5. clear H.
destruct c; simpl in *.
(* tag *)
destruct TL; try discriminate.
inversions H0; clear H0.
inversions H2; clear H2.
inversions H7; clear H7.
simpl in H5.
inversions H2; clear H2.
destruct TL; try discriminate.
inversions H9; clear H9.
puts (binds_func H0 B). inversions H; clear H H0.
puts (proj1 (proj1 H7)). simpl in H.
destruct H; rewrite* H.
(* Ksum *)
inversions TypA; clear TypA; try discriminate.
inversions H9; clear H9; try discriminate.
esplit; esplit; reflexivity.
(* matches *)
case_eq (cut (length TL) (map Delta.matches_arg (seq 2 (length l))));
introv R1.
assert (length TL <= length (map Delta.matches_arg (seq 2 (length l)))).
rewrite map_length, seq_length.
rewrite <- (list_forall2_length TypA). rewrite <- (proj1 Hvl).
omega.
destruct (cut_ok _ H R1). clear H. rewrite H5 in H0.
rewrite fold_right_app in H0.
destruct* (fold_arrow_eq _ _ _ _ _ H0).
destruct l1; discriminate.
(* record *)
case_eq (cut (length TL) (map typ_bvar (seq 1 (length l)))); introv R1.
assert (length TL <= length (map typ_bvar (seq 1 (length l)))).
rewrite map_length, seq_length.
rewrite <- (list_forall2_length TypA). rewrite <- (proj1 Hvl).
omega.
destruct (cut_ok _ H R1). clear H. rewrite H5 in H0.
rewrite fold_right_app in H0.
destruct* (fold_arrow_eq _ _ _ _ _ H0).
destruct l1; try discriminate.
inversions H2; clear H2. simpl in H; subst.
inversions H9; clear H9.
puts (binds_func H8 B). inversions H; clear H H8.
puts (proj1 (proj1 H10)). simpl in H.
destruct H; rewrite~ H.
(* Kprod *)
rewrite <- app_nil_end in H5.
exists l. exists n. exists vl.
split2*. subst. length_hyps. rewrite seq_length in *. omega.
(* sub *)
destruct Hvl.
inversions TypA; try discriminate.
(* recf *)
destruct Hvl.
inversions TypA; clear TypA; try discriminate.
inversions H0; clear H0.
inversions H6; clear H6; try discriminate.
destruct i; discriminate.
Qed.
Lemma fold_right_app_end : forall T1 T2 TL,
fold_right typ_arrow (typ_arrow T1 T2) TL =
fold_right typ_arrow T2 (TL ++ T1 :: nil).
Proof.
intros; rewrite* fold_right_app.
Qed.
Lemma in_matches_types : forall i l TL Us,
i < length l ->
In (nth i l 0, nth i Us typ_def)
(map_snd (fun T => typ_open T (TL ++ Us))
(combine l (map typ_bvar (seq (length TL) (length l))))).
Proof.
intros.
remember (0 + i) as m.
pattern i at 2.
replace i with (0+i) by reflexivity. rewrite <- Heqm.
generalize 0 at 1. intro d.
replace (length TL) with (0 + length TL) by auto.
gen i; generalize 0; induction l; intros.
simpl in H; elimtype False; omega.
destruct i; simpl.
replace m with n; auto. rewrite app_nth2.
replace (n + length TL - length TL) with n by omega.
auto. omega. omega.
right*.
apply (IHl (S n) i). simpl in H. omega.
omega.
Qed.
Lemma delta_typed_matches : forall l n, delta_typed_cst (@Const.matches l n).
Proof.
intros; intro; introv Typ0 TypA.
inversions Typ0; try discriminate; clear Typ0.
clear H1 H4.
unfold sch_open in H0; simpl in H0.
rewrite fold_right_app_end in H0.
poses Hlen (proj1 vl). simpl in Hlen.
destruct (fold_arrow_eq _ _ _ _ _ H0) as [HT HA]; clear H0.
autorewrite with list. simpl*.
forward~ (list_forall2_nth trm_def typ_def TypA (n:=length l)) as Typn.
rewrite* <- Hlen.
forward~ (list_forall2_nth typ_def typ_def HA (n:=length l)) as Eqn.
autorewrite with list. simpl. omega.
rewrite app_nth2 in Eqn by (rewrite map_length, seq_length; auto).
autorewrite with list in Eqn.
rewrite <- minus_n_n in Eqn.
simpl in Eqn.
destruct H5 as [_ WUs]. simpl in WUs.
inversions WUs; clear WUs.
inversions H3; clear H3.
simpl in Eqn.
inversions H1; clear H1 H2.
rewrite <- H6 in *; clear H6.
puts (value_fvar_is_const Typn H0).
assert (Cstr.cstr_sort (kind_cstr k') = Cstr.Ksum).
destruct H4 as [[HE _] _]. simpl in HE. destruct* HE.
clear -H1. destruct k'.
elimtype False. simpl in H1.
destruct* kind_valid0.
rewrite H1 in H; clear H1.
destruct H as [v [t2 Hv]].
apply* (list_forall_out (proj2 vl)). apply* nth_In. omega.
simpl.
rewrite Hv in *; clear Hv.
inversions Typn; clear Typn; try discriminate.
rewrite gc_lower_false in *.
inversions H6; clear H6; try discriminate.
destruct H11.
destruct H as [H _]. simpl in H.
do 3 (destruct Us; try discriminate). clear H.
simpl in *.
inversions H1; clear H1 H9 H10.
inversions H12; clear H12 H7.
inversions H3; clear H3 H9.
puts (binds_func H6 H0).
inversions H; clear H H0 H6 H5.
destruct H4; destruct H7. destruct H; destruct H1.
destruct k' as [[ks kcl kch] kv kr kh]; simpl in *.
destruct* kch.
assert (Hvt: In (v,t) kr) by auto*. clear H2 H1.
unfold Cstr.valid in kv; simpl in kv.
assert (Hvl: In v l) by auto*.
case_eq (index eq_nat_dec 0 v l); introv R2;
try elim (index_none_notin _ _ _ _ R2 Hvl).
destruct (index_ok _ 0 _ _ R2).
unfold Cstr.attr in *.
forward~ (list_forall2_nth trm_def typ_def TypA (n:=n0)) as Typv.
omega.
forward~ (list_forall2_nth typ_def typ_def HA (n:=n0)) as Eqv.
rewrite (list_forall2_length HA).
rewrite <- (list_forall2_length TypA).
omega.
rewrite <- Eqv in Typv; clear HA TypA Eqv.
rewrite app_nth1 in Typv by (rewrite map_length, seq_length; auto).
rewrite (map_nth _ 0) in Typv by rewrite* seq_length.
rewrite seq_nth in Typv by auto.
simpl in Typv.
rewrite (kh v (nth n0 lb0 typ_def) t) in Typv; auto.
eapply typing_app; rewrite* gc_lower_false.
unfold Cstr.unique. simpl. apply* set_mem_correct2.
unfold set_In; auto*.
rewrite <- H2; apply H0.
apply* (@in_matches_types n0 l (typ_fvar x :: b0 :: nil) lb0).
Qed.
Lemma delta_typed_record : forall l n, delta_typed_cst (@Const.record l n).
Proof.
intros; intro; introv Typ0 TypA.
elimtype False.
inversions Typ0; try discriminate.
puts (proj1 vl).
simpl in H.
unfold sch_open in H0. simpl in H0.
destruct TL using rev_ind. inversion TypA. subst; discriminate.
clear IHTL. rewrite fold_right_app in H0.
puts (fold_arrow_eq _ _ _ _ _ H0).
rewrite map_length, seq_length in H2.
puts (list_forall2_length TypA). rewrite app_length in H3; simpl in H3.
destruct H2. omega.
destruct H5.
simpl in H7. inversions H7; clear H7.
inversions H10; clear H10. discriminate.
Qed.
Lemma delta_typed_sub : forall a, delta_typed_cst (Const.sub a).
Proof.
intros; intro; introv Typ0 TypA.
inversions Typ0; clear Typ0; try discriminate.
puts (proj1 vl). simpl in H.
do 2 (destruct tl; try discriminate).
inversions TypA; clear TypA H H1 H4.
inversions H8; clear H8.
unfold sch_open in H0; simpl in H0.
destruct H5.
inversions H1; clear H1.
inversions H7; clear H7 H4.
inversions H8; clear H8; simpl in *.
inversions H0; clear H0 H.
inversions H3; clear H3.
puts (value_fvar_is_const H6 H0).
assert (Cstr.cstr_sort (kind_cstr k') = Cstr.Kprod).
destruct H2 as [[HE _] _]. simpl in HE. destruct* HE.
clear -H1. destruct k'.
elimtype False. simpl in H1.
destruct* kind_valid0.
rewrite H1 in H; clear H1.
destruct* H as [l' [nd [tl [HE HL]]]].
destruct vl. inversions* H1.
clear vl; subst.
rewrite Delta.record_args_ok. rewrite <- app_nil_end.
destruct (fold_app_inv _ _ H6) as [TL [Typ0 TypA]]; clear H6.
inversions Typ0; clear Typ0; try discriminate.
unfold sch_open in H1; simpl in H1; clear H3 H6.
destruct (fold_arrow_eq _ _ _ _ _ H1) as [Hx HTL].
rewrite map_length, seq_length. auto.
clear H1; destruct H7 as [_ WS].
simpl in WS.
inversions WS; clear WS.
simpl in Hx; subst.
inversions H3; clear H5 H3.
puts (binds_func H6 H0).
inversions H; clear H H0 H6.
puts (proj33 (proj1 H7)).
puts (proj32 (proj1 H2)).
destruct k' as [[ks kl km] kv kr kh]; simpl in *.
destruct* km.
puts (proj2 kv). simpl in H1.
assert (In a l') by auto. clear H H1.
case_eq (index eq_nat_dec 0 a l'); introv R1;
[|elim (index_none_notin _ _ _ _ R1 H3)].
destruct (index_ok _ 0 _ _ R1).
forward~ (list_forall2_nth Delta.trm_default typ_def TypA (n:=n)) as Typ.
rewrite* HL.
rewrite (kh a T (nth n TL typ_def)). auto.
unfold Cstr.unique. simpl. apply* set_mem_correct2. apply* H0.
apply* (proj2 H2).
forward~ (list_forall2_nth typ_def typ_def HTL (n:=n)) as Eqv.
rewrite map_length, seq_length; auto.
rewrite <- Eqv; clear Eqv.
rewrite (map_nth _ 0) by rewrite* seq_length.
rewrite* seq_nth.
apply (proj2 H7).
simpl.
rewrite <- H1.
apply* (@in_matches_types n l' (typ_fvar x::nil) lb).
Qed.
Lemma delta_typed : forall c tl vl K E gc T,
K ; E |(false,gc)|= const_app c tl ~: T ->
K ; E |(false,gc)|= @Delta.reduce c tl vl ~: T.
Proof.
intros.
destruct (fold_app_inv _ _ H) as [TL [Typ0 TypA]]; clear H.
destruct c.
apply* delta_typed_tag.
apply* delta_typed_matches.
apply* delta_typed_record.
apply* delta_typed_sub.
(* recf *)
puts (proj1 vl).
inversions TypA; clear TypA; try discriminate.
inversions H1; clear H1; try discriminate.
inversions H3; clear H3; try discriminate.
simpl in *.
inversions Typ0; try discriminate.
eapply typing_app; rewrite* gc_lower_false.
eapply typing_app; rewrite* gc_lower_false.
eapply typing_app; rewrite* gc_lower_false.
Qed.
End SndHyp.
Module Sound3 := Sound2.Mk3(SndHyp).