-
Notifications
You must be signed in to change notification settings - Fork 85
/
Copy pathpanPropsScript.sml
954 lines (906 loc) · 32.3 KB
/
panPropsScript.sml
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
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
(*
panLang Properties
*)
open preamble
panLangTheory panSemTheory
pan_commonPropsTheory;
val _ = new_theory"panProps";
val _ = set_grammar_ancestry ["panLang","panSem", "pan_commonProps"];
Definition v2word_def:
v2word (ValWord v) = Word v
End
Theorem length_flatten_eq_size_of_shape:
!v.
LENGTH (flatten v) = size_of_shape (shape_of v)
Proof
ho_match_mp_tac flatten_ind >> rw []
>- (cases_on ‘w’ >> fs [shape_of_def, flatten_def, size_of_shape_def]) >>
fs [shape_of_def, flatten_def, size_of_shape_def] >>
fs [LENGTH_FLAT, MAP_MAP_o] >> fs[SUM_MAP_FOLDL] >>
match_mp_tac FOLDL_CONG >> fs []
QED
Theorem mem_load_some_shape_eq:
∀sh adr dm (m: 'a word -> 'a word_lab) v.
mem_load sh adr dm m = SOME v ==>
shape_of v = sh
Proof
qsuff_tac ‘(∀sh adr dm (m: 'a word -> 'a word_lab) v.
mem_load sh adr dm m = SOME v ==> shape_of v = sh) /\
(∀sh adr dm (m: 'a word -> 'a word_lab) v.
mem_loads sh adr dm m = SOME v ==> MAP shape_of v = sh)’
>- metis_tac [] >>
ho_match_mp_tac mem_load_ind >> rw [mem_load_def] >>
cases_on ‘sh’ >> fs [option_case_eq] >>
rveq >> TRY (cases_on ‘m adr’) >> fs [shape_of_def] >>
metis_tac []
QED
Theorem flookup_res_var_some_eq_lookup:
FLOOKUP (res_var lc (v,FLOOKUP lc' v)) v = SOME value ==>
FLOOKUP lc' v = SOME value
Proof
rw [] >> cases_on ‘FLOOKUP lc' v’ >>
fs [res_var_def, FLOOKUP_UPDATE]
QED
Theorem flookup_res_var_diff_eq_org:
n <> m ==>
FLOOKUP (res_var lc (n,v)) m = FLOOKUP lc m
Proof
rw [] >> cases_on ‘v’ >>
fs [res_var_def, FLOOKUP_UPDATE, DOMSUB_FLOOKUP_NEQ]
QED
Theorem FLOOKUP_pan_res_var_thm:
FLOOKUP (panSem$res_var l (m,v)) n = if n = m then v else FLOOKUP l n
Proof
simp[oneline panSemTheory.res_var_def] >>
PURE_FULL_CASE_TAC >>
rw[DOMSUB_FLOOKUP_THM,FLOOKUP_UPDATE]
QED
Theorem list_rel_length_shape_of_flatten:
!vshs args.
LIST_REL (λvsh arg. SND vsh = shape_of arg) vshs args ==>
size_of_shape (Comb (MAP SND vshs)) = LENGTH (FLAT (MAP flatten args))
Proof
Induct >> rpt gen_tac >> strip_tac
>- (cases_on ‘args’ >> fs [size_of_shape_def]) >>
cases_on ‘args’ >> fs [] >> rveq >>
fs [size_of_shape_def] >>
last_x_assum (qspecl_then [‘t’] mp_tac) >>
fs [] >> last_x_assum (assume_tac o GSYM) >>
fs [] >>
fs [length_flatten_eq_size_of_shape]
QED
Theorem length_with_shape_eq_shape:
!sh ns.
LENGTH ns = size_of_shape (Comb sh) ==>
LENGTH sh = LENGTH (with_shape sh ns)
Proof
Induct >> rw [] >>
fs [with_shape_def] >>
fs [size_of_shape_def]
QED
Theorem fdoms_eq_flookup_some_none:
!fm fm' n v v'. FDOM fm = FDOM fm' /\
FLOOKUP fm n = SOME v ==> ?v. FLOOKUP fm' n = SOME v
Proof
rw [] >>
fs [flookup_thm] >> rveq >> fs [] >>
rfs []
QED
Theorem all_distinct_with_shape:
!sh ns n.
ALL_DISTINCT ns /\ n < LENGTH sh /\
LENGTH ns = size_of_shape (Comb sh) ==>
ALL_DISTINCT (EL n (with_shape sh ns))
Proof
Induct >> rw [] >>
fs [with_shape_def] >>
cases_on ‘n’ >> fs []
>- (
fs [size_of_shape_def] >>
‘size_of_shape h <= LENGTH ns’ by DECIDE_TAC >>
drule all_distinct_take >> fs []) >>
last_x_assum (qspecl_then [‘DROP (size_of_shape h) ns’, ‘n'’] mp_tac) >>
impl_tac
>- (
fs [size_of_shape_def] >>
‘size_of_shape h <= LENGTH ns’ by DECIDE_TAC >>
drule all_distinct_drop >> fs []) >> fs []
QED
Theorem el_mem_with_shape:
!sh ns n x.
n < LENGTH (with_shape sh ns) /\
LENGTH ns = size_of_shape (Comb sh) /\
MEM x (EL n (with_shape sh ns)) ==>
MEM x ns
Proof
Induct >> rw [] >>
fs [with_shape_def] >>
cases_on ‘n’ >> fs []
>- (
fs [size_of_shape_def] >>
‘size_of_shape h <= LENGTH ns’ by DECIDE_TAC >> drule MEM_TAKE >> fs []) >>
fs [size_of_shape_def] >>
last_x_assum (qspecl_then [‘DROP (size_of_shape h) ns’, ‘n'’, ‘x’] mp_tac) >>
fs [] >>
strip_tac >> drule MEM_DROP_IMP >>
fs []
QED
Theorem mem_with_shape_length:
!sh ns n.
LENGTH ns = size_of_shape (Comb sh) ∧ n < LENGTH sh ==>
MEM (EL n (with_shape sh ns)) (with_shape sh ns)
Proof
Induct >> rw [] >>
cases_on ‘n’ >> fs [] >>
fs [with_shape_def] >>
disj2_tac >>
first_x_assum match_mp_tac >>
fs [size_of_shape_def]
QED
Theorem with_shape_el_take_drop_eq:
!sh ns n.
LENGTH ns = size_of_shape (Comb sh) ∧
n < LENGTH sh ==>
EL n (with_shape sh ns) =
TAKE (size_of_shape (EL n sh)) (DROP (size_of_shape (Comb (TAKE n sh))) ns)
Proof
Induct >> rw [] >>
cases_on ‘n’ >> fs []
>- fs [with_shape_def, size_of_shape_def] >>
fs [with_shape_def, size_of_shape_def] >>
last_x_assum (qspecl_then [‘DROP (size_of_shape h) ns’, ‘n'’] mp_tac) >>
impl_tac >- fs [] >>
strip_tac >> fs [DROP_DROP_T]
QED
Theorem all_distinct_with_shape_distinct:
!sh ns x y.
ALL_DISTINCT ns ∧ LENGTH ns = size_of_shape (Comb sh) ∧
MEM x (with_shape sh ns) ∧ MEM y (with_shape sh ns) ∧ x <> y ∧
x <> [] ∧ y <> [] ==>
DISJOINT (set x) (set y)
Proof
Induct >> rw [] >>
fs [with_shape_def]
>- (
cases_on ‘size_of_shape h = 0’ >> fs [] >>
‘x = y’ suffices_by fs [] >>
‘size_of_shape h <= LENGTH ns’ by
fs [size_of_shape_def] >>
qpat_x_assum ‘x ≠ y’ kall_tac >>
fs [TAKE])
>- (
fs [MEM_EL] >>
‘EL n (with_shape sh (DROP (size_of_shape h) ns)) =
TAKE (size_of_shape (EL n sh)) (DROP (size_of_shape (Comb (TAKE n sh)))
(DROP (size_of_shape h) ns))’ by (
match_mp_tac with_shape_el_take_drop_eq >>
fs [size_of_shape_def] >>
‘LENGTH (DROP (size_of_shape h) ns) = size_of_shape (Comb sh)’ by
fs [size_of_shape_def] >>
drule length_with_shape_eq_shape >> fs []) >>
fs [] >>
fs [DROP_DROP_T, DROP_TAKE] >>
match_mp_tac disjoint_take_drop_sum >>
fs [])
>- (
fs [MEM_EL] >>
‘EL n (with_shape sh (DROP (size_of_shape h) ns)) =
TAKE (size_of_shape (EL n sh)) (DROP (size_of_shape (Comb (TAKE n sh)))
(DROP (size_of_shape h) ns))’ by (
match_mp_tac with_shape_el_take_drop_eq >>
fs [size_of_shape_def] >>
‘LENGTH (DROP (size_of_shape h) ns) = size_of_shape (Comb sh)’ by
fs [size_of_shape_def] >>
drule length_with_shape_eq_shape >> fs []) >>
fs [] >>
fs [DROP_DROP_T, DROP_TAKE] >>
match_mp_tac disjoint_drop_take_sum >>
fs []) >>
last_x_assum (qspec_then ‘DROP (size_of_shape h) ns’ mp_tac) >>
disch_then (qspecl_then [‘x’,‘y’] mp_tac) >>
impl_tac
>- fs [ALL_DISTINCT_DROP, size_of_shape_def] >>
fs []
QED
Theorem all_distinct_disjoint_with_shape:
!sh ns n n'.
ALL_DISTINCT ns /\ n < LENGTH sh /\ n' < LENGTH sh /\
n <> n' /\
LENGTH ns = size_of_shape (Comb sh) ==>
DISJOINT (set (EL n (with_shape sh ns))) (set (EL n' (with_shape sh ns)))
Proof
Induct >> rw [] >>
fs [with_shape_def] >>
cases_on ‘n’ >> cases_on ‘n'’ >> fs []
>- (
fs [MEM_EL] >>
‘EL n (with_shape sh (DROP (size_of_shape h) ns)) =
TAKE (size_of_shape (EL n sh)) (DROP (size_of_shape (Comb (TAKE n sh)))
(DROP (size_of_shape h) ns))’ by (
match_mp_tac with_shape_el_take_drop_eq >>
fs [size_of_shape_def] >>
‘LENGTH (DROP (size_of_shape h) ns) = size_of_shape (Comb sh)’ by
fs [size_of_shape_def] >>
drule length_with_shape_eq_shape >> fs []) >>
fs [] >>
fs [DROP_DROP_T, DROP_TAKE] >>
match_mp_tac disjoint_take_drop_sum >>
fs [])
>- (
fs [MEM_EL] >>
‘EL n'' (with_shape sh (DROP (size_of_shape h) ns)) =
TAKE (size_of_shape (EL n'' sh)) (DROP (size_of_shape (Comb (TAKE n'' sh)))
(DROP (size_of_shape h) ns))’ by (
match_mp_tac with_shape_el_take_drop_eq >>
fs [size_of_shape_def] >>
‘LENGTH (DROP (size_of_shape h) ns) = size_of_shape (Comb sh)’ by
fs [size_of_shape_def] >>
drule length_with_shape_eq_shape >> fs []) >>
fs [] >>
fs [DROP_DROP_T, DROP_TAKE] >>
match_mp_tac disjoint_drop_take_sum >>
fs []) >>
last_x_assum match_mp_tac >>
fs [size_of_shape_def, ALL_DISTINCT_DROP]
QED
Theorem all_distinct_mem_zip_disjoint_with_shape:
LENGTH l = LENGTH sh /\ LENGTH sh = LENGTH (with_shape sh ns) /\
ALL_DISTINCT ns /\ LENGTH ns = size_of_shape (Comb sh) /\
MEM (x,a,xs) (ZIP (l,ZIP (sh,with_shape sh ns))) /\
MEM (y,b,ys) (ZIP (l,ZIP (sh,with_shape sh ns))) /\
x ≠ y ==>
DISJOINT (set xs) (set ys)
Proof
rw [] >>
‘LENGTH l = LENGTH (ZIP (sh,with_shape sh ns))’ by fs [] >>
drule MEM_ZIP >>
disch_then (qspec_then ‘(x,a,xs)’ assume_tac) >>
drule MEM_ZIP >>
disch_then (qspec_then ‘(y,b,ys)’ assume_tac) >>
fs [] >> rveq >>
cases_on ‘n = n'’ >> fs [] >>
drule EL_ZIP >> drule EL_ZIP >>
disch_then (qspec_then ‘n’ assume_tac) >>
disch_then (qspec_then ‘n'’ assume_tac) >>
rfs [] >> rveq >> fs [] >>
match_mp_tac all_distinct_disjoint_with_shape >>
fs []
QED
Theorem all_distinct_alist_no_overlap:
ALL_DISTINCT (ns:num list) /\
LENGTH ns = size_of_shape (Comb sh) ∧
LENGTH vs = LENGTH sh ⇒
no_overlap (alist_to_fmap (ZIP (vs,ZIP (sh,with_shape sh ns))))
Proof
rw [] >>
fs [no_overlap_def] >>
conj_tac
>- (
rw [] >>
drule ALOOKUP_MEM >>
strip_tac >> fs [] >>
drule length_with_shape_eq_shape >>
strip_tac >>
drule LENGTH_ZIP >>
strip_tac >> fs [] >>
‘LENGTH vs = LENGTH (ZIP (sh,with_shape sh ns))’ by fs [] >>
drule MEM_ZIP >>
disch_then (qspec_then ‘(x,a,xs)’ mp_tac) >>
strip_tac >> fs [] >> rveq >>
‘LENGTH sh = LENGTH (with_shape sh ns)’ by fs [] >>
drule EL_ZIP >>
disch_then (qspec_then ‘n’ mp_tac) >>
impl_tac >- fs [] >>
strip_tac >> fs [] >>
match_mp_tac all_distinct_with_shape >>
fs []) >>
rw [] >>
CCONTR_TAC >> fs [] >>
dxrule ALOOKUP_MEM >>
dxrule ALOOKUP_MEM >>
rpt strip_tac >>
drule length_with_shape_eq_shape >>
strip_tac >>
drule length_with_shape_eq_shape >>
drule (INST_TYPE [``:'b``|->``:num``] all_distinct_mem_zip_disjoint_with_shape) >>
disch_then (qspecl_then [‘ys’, ‘y’, ‘xs’, ‘x’, ‘ns’, ‘b’, ‘a’] assume_tac) >>
rfs []
QED
Theorem all_distinct_alist_ctxt_max:
ALL_DISTINCT (ns:num list) /\
LENGTH ns = size_of_shape (Comb sh) ∧
LENGTH vs = LENGTH sh ⇒
ctxt_max (list_max ns)
(alist_to_fmap (ZIP (vs,ZIP (sh,with_shape sh ns))))
Proof
rw [] >> fs [ctxt_max_def] >>
rw [] >>
‘MEM x ns’ suffices_by (
assume_tac list_max_max >>
pop_assum (qspec_then ‘ns’ assume_tac) >>
fs [EVERY_MEM]) >>
drule ALOOKUP_MEM >>
strip_tac >>
drule length_with_shape_eq_shape >>
strip_tac >>
drule LENGTH_ZIP >>
strip_tac >> fs [] >>
‘LENGTH vs = LENGTH (ZIP (sh,with_shape sh ns))’ by fs [] >>
drule MEM_ZIP >>
disch_then (qspec_then ‘(v,a,xs)’ mp_tac) >>
strip_tac >> fs [] >>
rveq >> ‘LENGTH sh = LENGTH (with_shape sh ns)’ by fs [] >>
drule EL_ZIP >>
disch_then (qspec_then ‘n’ mp_tac) >>
impl_tac >- fs [] >>
strip_tac >> fs [] >> rveq >>
drule el_mem_with_shape >>
fs []
QED
Theorem list_rel_flatten_with_shape_length:
!sh ns args v n.
LENGTH ns = LENGTH (FLAT (MAP flatten args))/\
size_of_shape (Comb sh) = LENGTH (FLAT (MAP flatten args)) /\
EL n args = v /\ n < LENGTH args /\ LENGTH args = LENGTH sh /\
LIST_REL (λsh arg. sh = shape_of arg) sh args ==>
LENGTH (EL n (with_shape sh ns)) = LENGTH (flatten v)
Proof
Induct >> rw []
>- fs [with_shape_def, size_of_shape_def] >>
fs [with_shape_def, size_of_shape_def] >>
cases_on ‘n’ >> fs []
>- fs [length_flatten_eq_size_of_shape] >>
last_x_assum match_mp_tac >>
‘LENGTH (flatten arg) = size_of_shape (shape_of arg)’ by
fs [length_flatten_eq_size_of_shape] >>
fs []
QED
Theorem list_rel_flatten_with_shape_flookup:
!sh ns args v n n'.
ALL_DISTINCT ns ∧ LENGTH ns = LENGTH (FLAT (MAP flatten args)) /\
size_of_shape (Comb sh) = LENGTH (FLAT (MAP flatten args)) /\
EL n args = v /\ n < LENGTH args /\ LENGTH args = LENGTH sh /\
LIST_REL (λsh arg. sh = shape_of arg) sh args /\
LENGTH (EL n (with_shape sh ns)) = LENGTH (flatten v) /\
n' < LENGTH (EL n (with_shape sh ns)) ==>
FLOOKUP (FEMPTY |++ ZIP (ns,FLAT (MAP flatten args)))
(EL n' (EL n (with_shape sh ns))) =
SOME (EL n' (flatten v))
Proof
Induct >> rw []
>- fs [with_shape_def, size_of_shape_def] >>
fs [with_shape_def, size_of_shape_def] >>
cases_on ‘n’ >> fs []
>- (
‘FLOOKUP (FEMPTY |++ ZIP (ns,flatten arg ++ FLAT (MAP flatten ys)))
(EL n' (TAKE (size_of_shape (shape_of arg)) ns)) =
SOME (EL n' (flatten arg ++ FLAT (MAP flatten ys)))’ by (
‘size_of_shape (shape_of arg) = LENGTH (flatten arg)’ by
fs [length_flatten_eq_size_of_shape] >>
fs [] >>
‘EL n' (flatten arg ++ FLAT (MAP flatten ys)) = EL n' (flatten arg)’ by (
match_mp_tac EL_APPEND1 >> fs []) >>
fs [] >>
‘FEMPTY |++ ZIP (TAKE (LENGTH (flatten arg)) ns ++ DROP (LENGTH (flatten arg)) ns,
flatten arg ++ FLAT (MAP flatten ys)) =
FEMPTY |++ ZIP (TAKE (LENGTH (flatten arg)) ns,flatten arg) |++
ZIP (DROP (LENGTH (flatten arg)) ns,FLAT (MAP flatten ys))’ by (
drule ZIP_APPEND >>
disch_then (qspecl_then [‘DROP (LENGTH (flatten arg)) ns’, ‘FLAT (MAP flatten ys)’]mp_tac) >>
impl_tac >- fs [] >>
strip_tac >> pop_assum (assume_tac o GSYM) >>
fs [] >>
fs [FUPDATE_LIST_APPEND]) >>
fs [] >> pop_assum kall_tac >>
‘FEMPTY |++ ZIP (TAKE (LENGTH (flatten arg)) ns,flatten arg) |++
ZIP (DROP (LENGTH (flatten arg)) ns,FLAT (MAP flatten ys)) =
FEMPTY |++
ZIP (DROP (LENGTH (flatten arg)) ns,FLAT (MAP flatten ys)) |++
ZIP (TAKE (LENGTH (flatten arg)) ns,flatten arg)’ by (
match_mp_tac FUPDATE_LIST_APPEND_COMMUTES >>
fs [MAP_ZIP] >> match_mp_tac all_distinct_take_frop_disjoint >> fs []) >>
fs [] >> pop_assum kall_tac >>
match_mp_tac update_eq_zip_flookup >>
fs [] >>
match_mp_tac all_distinct_take >>
fs []) >>
fs [] >>
pop_assum kall_tac >>
metis_tac [EL_APPEND1]) >>
‘FLOOKUP (FEMPTY |++ ZIP (ns,flatten arg ++ FLAT (MAP flatten ys)))
(EL n'
(EL n'' (with_shape sh (DROP (size_of_shape (shape_of arg)) ns)))) =
FLOOKUP (FEMPTY |++ ZIP (DROP (size_of_shape (shape_of arg)) ns,FLAT (MAP flatten ys)))
(EL n'
(EL n'' (with_shape sh (DROP (size_of_shape (shape_of arg)) ns))))’ by (
‘FEMPTY |++ ZIP (ns,flatten arg ++ FLAT (MAP flatten ys)) =
FEMPTY |++
ZIP (TAKE (LENGTH (flatten arg)) ns,flatten arg) |++
ZIP (DROP (LENGTH (flatten arg)) ns,FLAT (MAP flatten ys))’ by (
match_mp_tac fm_zip_append_take_drop >>
fs []) >>
fs [] >> pop_assum kall_tac >>
‘FLOOKUP
(FEMPTY |++ ZIP (TAKE (LENGTH (flatten arg)) ns,flatten arg))
(EL n'
(EL n'' (with_shape sh (DROP (size_of_shape (shape_of arg)) ns)))) = NONE’ by (
match_mp_tac not_mem_fst_zip_flookup_empty >>
fs [] >> drule all_distinct_take >> disch_then (qspec_then ‘LENGTH (flatten arg)’ assume_tac) >>
fs [] >>
CCONTR_TAC >> fs [] >>
fs [GSYM length_flatten_eq_size_of_shape] >>
‘TAKE (LENGTH (flatten arg)) ns =
EL 0 (with_shape (shape_of arg::sh) ns)’ by
fs [with_shape_def, length_flatten_eq_size_of_shape] >>
‘EL n'' (with_shape sh (DROP (LENGTH (flatten arg)) ns)) =
EL (SUC n'') (with_shape (shape_of arg::sh) ns)’ by
fs [with_shape_def, length_flatten_eq_size_of_shape] >>
drule all_distinct_disjoint_with_shape >>
disch_then (qspecl_then [‘shape_of arg::sh’, ‘SUC n''’, ‘0’] mp_tac) >>
impl_tac >- fs [length_flatten_eq_size_of_shape, size_of_shape_def] >>
strip_tac >> fs [] >> drule disjoint_not_mem_el >> metis_tac []) >>
drule fupdate_flookup_zip_elim >>
disch_then (qspecl_then [‘DROP (LENGTH (flatten arg)) ns’, ‘FLAT (MAP flatten ys)’] mp_tac) >>
impl_tac >- (fs [] >> match_mp_tac all_distinct_take >> fs []) >>
fs [] >> strip_tac >>
fs [length_flatten_eq_size_of_shape]) >>
fs [] >>
pop_assum kall_tac >>
last_x_assum (qspecl_then [‘DROP (size_of_shape (shape_of arg)) ns’,
‘ys’, ‘n''’, ‘n'’] mp_tac) >>
impl_tac >- fs [ALL_DISTINCT_DROP, GSYM length_flatten_eq_size_of_shape] >> fs []
QED
Theorem eval_upd_clock_eq:
!t e ck. eval (t with clock := ck) e = eval t e
Proof
ho_match_mp_tac eval_ind >> rw [] >>
fs [eval_def] >>
qsuff_tac ‘OPT_MMAP (λa. eval (t with clock := ck) a) es =
OPT_MMAP (λa. eval t a) es’ >>
fs [] >>
pop_assum mp_tac >>
qid_spec_tac ‘es’ >>
Induct >> rw [] >>
fs [OPT_MMAP_def]
QED
Theorem opt_mmap_eval_upd_clock_eq:
!es s ck. OPT_MMAP (eval (s with clock := ck + s.clock)) es =
OPT_MMAP (eval s) es
Proof
rw [] >>
match_mp_tac IMP_OPT_MMAP_EQ >>
fs [MAP_EQ_EVERY2, LIST_REL_EL_EQN] >>
rw [] >>
metis_tac [eval_upd_clock_eq]
QED
Theorem opt_mmap_eval_upd_clock_eq1:
!es s ck. OPT_MMAP (eval (s with clock := ck)) es =
OPT_MMAP (eval s) es
Proof
rw [] >>
match_mp_tac IMP_OPT_MMAP_EQ >>
fs [MAP_EQ_EVERY2, LIST_REL_EL_EQN] >>
rw [] >>
metis_tac [eval_upd_clock_eq]
QED
Theorem evaluate_add_clock_eq:
!p t res st ck.
evaluate (p,t) = (res,st) /\ res <> SOME TimeOut ==>
evaluate (p,t with clock := t.clock + ck) = (res,st with clock := st.clock + ck)
Proof
recInduct evaluate_ind >> rw []
>~ [‘While’]
>- (once_rewrite_tac [evaluate_def] >>
qpat_x_assum ‘evaluate _ = _’ mp_tac >>
rw[Once evaluate_def] >>
gvs[eval_upd_clock_eq,AllCaseEqs()] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs(),dec_clock_def]) >>
gvs[evaluate_def,AllCaseEqs(),eval_upd_clock_eq] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[oneline nb_op_def,AllCaseEqs(),
oneline sh_mem_load_def,
oneline sh_mem_store_def,
set_var_def,
empty_locals_def,
dec_clock_def,
opt_mmap_eval_upd_clock_eq
]
QED
Theorem evaluate_clock_sub:
!p t res st ck.
evaluate (p,t) = (res,st with clock := st.clock + ck) ∧
res <> SOME TimeOut ⇒
evaluate (p,t with clock := t.clock - ck) = (res,st)
Proof
(* TODO: generated names *)
recInduct evaluate_ind >> rw []
>~ [‘While’]
>- (once_rewrite_tac [evaluate_def] >>
qpat_x_assum ‘evaluate _ = _’ mp_tac >>
rw[Once evaluate_def] >>
gvs[eval_upd_clock_eq,AllCaseEqs()] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs(),dec_clock_def] >>
imp_res_tac evaluate_clock >>
gvs[] >>
rw[state_component_equality] >>
first_x_assum $ resolve_then (Pos hd) mp_tac EQ_REFL >>
rw[] >>
last_x_assum $ qspecl_then [‘s1' with clock := s1'.clock - ck’,‘ck’] mp_tac >>
(impl_tac >- rw[state_component_equality]) >>
strip_tac >>
gvs[])
>~ [‘Seq’]
>- (once_rewrite_tac [evaluate_def] >>
qpat_x_assum ‘evaluate _ = _’ mp_tac >>
rw[Once evaluate_def] >>
gvs[eval_upd_clock_eq,AllCaseEqs()] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs(),dec_clock_def] >>
imp_res_tac evaluate_clock >>
gvs[] >>
rw[state_component_equality] >>
first_x_assum $ resolve_then (Pos hd) mp_tac EQ_REFL >>
rw[] >>
last_x_assum $ qspecl_then [‘s1' with clock := s1'.clock - ck’,‘ck’] mp_tac >>
(impl_tac >- rw[state_component_equality]) >>
strip_tac >>
gvs[])
>~ [‘Dec’]
>- (once_rewrite_tac [evaluate_def] >>
qpat_x_assum ‘evaluate _ = _’ mp_tac >>
rw[Once evaluate_def] >>
gvs[eval_upd_clock_eq,AllCaseEqs()] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs(),dec_clock_def] >>
imp_res_tac evaluate_clock >>
gvs[]
>- rw[state_component_equality] >>
last_x_assum $ qspecl_then [‘st'' with clock := st''.clock - ck’,‘ck’] mp_tac >>
(impl_tac >- gvs[state_component_equality]) >>
strip_tac >>
gvs[state_component_equality])
>~ [‘Call’]
>- (gvs[evaluate_def,AllCaseEqs(),eval_upd_clock_eq,opt_mmap_eval_upd_clock_eq1,dec_clock_def,
empty_locals_def,set_var_def] >>
imp_res_tac evaluate_clock >>
gvs[empty_locals_def] >>
TRY $ first_x_assum $ irule_at $ Pat ‘evaluate _ = _’ >>
gvs[state_component_equality] >>
TRY $ first_x_assum $ irule_at $ Pat ‘evaluate _ = _’ >>
rw[] >>
qrefine ‘_ with locals := (_:('a,'b) state).locals’ >>
rw[] >>
gvs[] >>
metis_tac[])
>~ [‘DecCall’]
>- (gvs[evaluate_def,AllCaseEqs(),eval_upd_clock_eq,opt_mmap_eval_upd_clock_eq1,dec_clock_def,
empty_locals_def,set_var_def] >>
rpt(pairarg_tac >> gvs[]) >>
imp_res_tac evaluate_clock >>
gvs[empty_locals_def] >>
TRY $ first_x_assum $ irule_at $ Pat ‘evaluate _ = _’ >>
gvs[state_component_equality] >>
TRY $ first_x_assum $ irule_at $ Pat ‘evaluate _ = _’ >>
rw[]
>~ [‘UNCURRY’]
>- (qexists_tac ‘st' with clock := st'.clock - ck’ >>
simp[] >>
last_x_assum $ qspec_then ‘st with <|locals := st''.locals|>’ $ dep_rewrite.DEP_ONCE_REWRITE_TAC o single >>
simp[] >>
rw[state_component_equality]) >>
qrefine ‘_ with locals := (_:('a,'b) state).locals’ >>
rw[] >>
gvs[] >>
metis_tac[]) >>
gvs[evaluate_def,state_component_equality,AllCaseEqs(),eval_upd_clock_eq,
oneline nb_op_def,oneline sh_mem_load_def,
oneline sh_mem_store_def, set_var_def, empty_locals_def,
dec_clock_def,opt_mmap_eval_upd_clock_eq1
] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[state_component_equality]
QED
Theorem evaluate_io_events_mono:
!exps s1 res s2.
evaluate (exps,s1) = (res, s2)
⇒
s1.ffi.io_events ≼ s2.ffi.io_events
Proof
recInduct evaluate_ind >>
rw []
>~ [‘While’]
>- (pop_assum mp_tac >>
simp[Once evaluate_def] >> strip_tac >>
gvs[AllCaseEqs(),empty_locals_def] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs(),dec_clock_def] >>
imp_res_tac IS_PREFIX_TRANS) >>
gvs[evaluate_def,AllCaseEqs(),
oneline nb_op_def,oneline sh_mem_load_def,
oneline sh_mem_store_def, set_var_def, empty_locals_def,
dec_clock_def,opt_mmap_eval_upd_clock_eq1,
ffiTheory.call_FFI_def] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs()] >>
imp_res_tac IS_PREFIX_TRANS
QED
Theorem evaluate_add_clock_io_events_mono:
∀exps s extra.
(SND(evaluate(exps,s))).ffi.io_events ≼
(SND(evaluate(exps,s with clock := s.clock + extra))).ffi.io_events
Proof
‘∀exps s extra res s'.
evaluate(exps,s) = (res,s') ⇒
s'.ffi.io_events ≼ (SND(evaluate(exps,s with clock := s.clock + extra))).ffi.io_events’
suffices_by metis_tac [FST,SND,PAIR] >>
recInduct evaluate_ind >>
rw []
>~ [‘While’]
>- (simp[Once evaluate_def] >>
pop_assum mp_tac >>
simp[Once evaluate_def] >>
strip_tac >>
gvs[AllCaseEqs(),
eval_upd_clock_eq,
empty_locals_def,
dec_clock_def]
>- (IF_CASES_TAC >> gvs[] >>
pairarg_tac >>
gvs[] >>
rpt(PURE_TOP_CASE_TAC >> gvs[]) >>
metis_tac[FST,SND,PAIR,evaluate_io_events_mono,IS_PREFIX_TRANS,
Q.prove(‘(x with clock := y).ffi = x.ffi’,simp[])]) >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs()]
>~ [‘evaluate _ = (SOME TimeOut, _)’]
>- (rpt(PURE_TOP_CASE_TAC >> gvs[]) >>
metis_tac[FST,SND,PAIR,evaluate_io_events_mono,IS_PREFIX_TRANS,
Q.prove(‘(x with clock := y).ffi = x.ffi’,simp[])]) >>
qpat_x_assum ‘evaluate (_,_ with <|clock := _.clock − 1|>) = _’ assume_tac >>
drule_then (qspec_then ‘extra’ assume_tac) evaluate_add_clock_eq >>
gvs[])
>~ [‘Seq’]
>- (gvs[evaluate_def,AllCaseEqs(),
oneline nb_op_def,oneline sh_mem_load_def,
oneline sh_mem_store_def, set_var_def, empty_locals_def,
dec_clock_def,opt_mmap_eval_upd_clock_eq1,
eval_upd_clock_eq,ffiTheory.call_FFI_def] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs()]
>- (drule_then (qspec_then ‘extra’ assume_tac) evaluate_add_clock_eq >>
gvs[]) >>
rw[] >>
metis_tac[FST,SND,PAIR,evaluate_io_events_mono,IS_PREFIX_TRANS])
>~ [‘Call’]
>- (gvs[evaluate_def,AllCaseEqs(),
oneline nb_op_def,oneline sh_mem_load_def,
oneline sh_mem_store_def, set_var_def, empty_locals_def,
dec_clock_def,opt_mmap_eval_upd_clock_eq1,
eval_upd_clock_eq,ffiTheory.call_FFI_def] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs()] >>
rw[] >>
rpt(IF_CASES_TAC ORELSE PURE_TOP_CASE_TAC >> gvs[]) >>
imp_res_tac evaluate_io_events_mono >> gvs[] >>
TRY(qpat_x_assum ‘evaluate (_,_ with <|locals := _; clock := _.clock − 1|>) = _’ assume_tac >>
drule_then (qspec_then ‘extra’ assume_tac) evaluate_add_clock_eq >>
gvs[]) >>
metis_tac[FST,SND,PAIR,evaluate_io_events_mono,IS_PREFIX_TRANS,
Q.prove(‘(x with locals := y).ffi = x.ffi’,simp[])])
>~ [‘DecCall’]
>- (gvs[evaluate_def,AllCaseEqs(),
oneline nb_op_def,oneline sh_mem_load_def,
oneline sh_mem_store_def, set_var_def, empty_locals_def,
dec_clock_def,opt_mmap_eval_upd_clock_eq1,
eval_upd_clock_eq,ffiTheory.call_FFI_def] >>
rpt(pairarg_tac >> gvs[]) >>
gvs[AllCaseEqs()] >>
rw[] >>
rpt(IF_CASES_TAC ORELSE PURE_TOP_CASE_TAC >> gvs[]) >>
imp_res_tac evaluate_io_events_mono >> gvs[] >>
rpt(pairarg_tac >> gvs[]) >>
TRY(qpat_x_assum ‘evaluate (_,_ with <|locals := _; clock := _.clock − 1|>) = _’ assume_tac >>
drule_then (qspec_then ‘extra’ assume_tac) evaluate_add_clock_eq >>
gvs[]) >>
metis_tac[FST,SND,PAIR,evaluate_io_events_mono,IS_PREFIX_TRANS,
Q.prove(‘(x with locals := y).ffi = x.ffi’,simp[])]
) >>
gvs[evaluate_def,AllCaseEqs(),
oneline nb_op_def,oneline sh_mem_load_def,
oneline sh_mem_store_def, set_var_def, empty_locals_def,
dec_clock_def,opt_mmap_eval_upd_clock_eq1,
eval_upd_clock_eq,ffiTheory.call_FFI_def] >>
rpt(pairarg_tac >> gvs[]) >>
rw[] >>
gvs[AllCaseEqs()] >>
metis_tac[FST,SND,PAIR,IS_PREFIX_TRANS]
QED
Theorem update_locals_not_vars_eval_eq:
∀s e v n w.
~MEM n (var_exp e) /\
eval s e = SOME v ==>
eval (s with locals := s.locals |+ (n,w)) e = SOME v
Proof
ho_match_mp_tac eval_ind >>
rpt conj_tac >> rpt gen_tac >> strip_tac
>- fs [eval_def]
>- fs [eval_def, var_exp_def, FLOOKUP_UPDATE]
>- fs [eval_def]
>- (
rpt gen_tac >>
fs [var_exp_def] >>
strip_tac >>
rpt (pop_assum mp_tac) >>
MAP_EVERY qid_spec_tac [‘s’, ‘n’, ‘v’, ‘es’] >>
Induct >> rw []
>- gs [eval_def, OPT_MMAP_def] >>
gs [eval_def, OPT_MMAP_def] >>
every_case_tac >> gvs []
>- (
first_x_assum (qspec_then ‘h’ mp_tac) >>
impl_tac >- gs [] >>
strip_tac >> gs [])
>- (
last_x_assum (qspecl_then [‘Struct t’, ‘n’, ‘s’] mp_tac) >>
impl_tac >- metis_tac [] >>
strip_tac >> gs []) >>
conj_asm1_tac
>- (
first_x_assum (qspec_then ‘h’ mp_tac) >>
impl_tac >- gs [] >>
strip_tac >> rgs []) >>
gvs [] >>
last_x_assum (qspecl_then [‘Struct t'’, ‘n’, ‘s’] mp_tac) >>
impl_tac >- metis_tac [] >>
simp[])
>- (
rpt gen_tac >>
strip_tac >>
fs [var_exp_def, eval_def] >>
cases_on ‘eval s e’ >>
fs [])
>- (
rpt gen_tac >>
strip_tac >> fs [var_exp_def] >>
fs [eval_def, CaseEq "option", CaseEq "word_lab"] >>
rveq >> fs [mem_load_def])
>- (
rpt gen_tac >>
strip_tac >> fs [var_exp_def] >>
fs [eval_def, CaseEq "option", CaseEq "word_lab"] >>
rveq >> fs [mem_load_def])
>- (
rpt gen_tac >>
strip_tac >> fs [var_exp_def, ETA_AX] >>
fs [eval_def, CaseEq "option", ETA_AX] >>
qexists_tac ‘ws’ >>
fs [opt_mmap_eq_some, ETA_AX,
MAP_EQ_EVERY2, LIST_REL_EL_EQN] >>
rw [] >>
fs [MEM_FLAT, MEM_MAP] >>
metis_tac [EL_MEM])
>- (
rpt gen_tac >>
strip_tac >>
gvs [var_exp_def, eval_def, AllCaseEqs(),opt_mmap_eq_some,SF DNF_ss,
DefnBase.one_line_ify NONE pan_op_def,MAP_EQ_CONS,MEM_FLAT,MEM_MAP,PULL_EXISTS] >>
metis_tac[]
)
>- (
rw [] >>
gs [var_exp_def, eval_def] >>
every_case_tac >> gvs []) >>
rw [] >>
gs [var_exp_def, eval_def] >>
every_case_tac >> gvs []
QED
Theorem write_bytearray_update_byte:
∀bytes ad ad' m adrs be.
byte_aligned ad ∧
(∃w. m ad = Word w) ⇒
∃w.
write_bytearray ad' bytes m adrs be
ad = Word w
Proof
Induct >>
rw [] >>
gs [panSemTheory.write_bytearray_def] >>
TOP_CASE_TAC >> gs [] >>
gs [mem_store_byte_def] >>
every_case_tac >> gs [] >>
rveq >> gs [] >>
gs [byte_align_aligned] >>
fs [APPLY_UPDATE_THM] >>
every_case_tac >> gs [] >>
fs [APPLY_UPDATE_THM]
QED
Theorem evaluate_clock_sub1:
!p t res st t' ck.
evaluate (p,t) = (res,st) /\ res <> SOME TimeOut ∧
evaluate (p,t with clock := ck + t.clock) =
evaluate (p,t') ⇒
evaluate (p,t) = evaluate (p,t' with clock := t'.clock - ck)
Proof
rw [] >> gs [] >>
last_x_assum assume_tac >>
drule evaluate_add_clock_eq >>
disch_then (qspec_then ‘ck’ mp_tac) >>
gs [] >>
strip_tac >>
qpat_x_assum ‘_ = evaluate (p,t')’ kall_tac >>
once_rewrite_tac [EQ_SYM_EQ] >>
match_mp_tac evaluate_clock_sub >>
gs []
QED
Theorem evaluate_invariants:
∀p t res st.
evaluate (p,t) = (res,st) ⇒
st.memaddrs = t.memaddrs ∧ st.sh_memaddrs = t.sh_memaddrs ∧
st.be = t.be ∧ st.eshapes = t.eshapes ∧ st.base_addr = t.base_addr ∧
st.code = t.code
Proof
Ho_Rewrite.PURE_REWRITE_TAC[FORALL_AND_THM,IMP_CONJ_THM] >> rpt conj_tac >>
recInduct evaluate_ind >>
(rw[Once evaluate_def]
>~ [‘While’]
>- (qpat_x_assum ‘evaluate _ = _’ (strip_assume_tac o ONCE_REWRITE_RULE[evaluate_def]) >>
gvs[AllCaseEqs(),empty_locals_def,ELIM_UNCURRY,dec_clock_def] >>
metis_tac[PAIR,FST,SND])
>~[‘ShMemLoad’]
>- (Cases_on ‘op’>>
gvs[Once evaluate_def,AllCaseEqs(),ELIM_UNCURRY,empty_locals_def,
dec_clock_def,set_var_def,nb_op_def,sh_mem_store_def,
sh_mem_load_def] >>
metis_tac[PAIR,FST,SND])
>~[‘ShMemStore’]
>- (Cases_on ‘op’>>
gvs[Once evaluate_def,AllCaseEqs(),ELIM_UNCURRY,empty_locals_def,
dec_clock_def,set_var_def,nb_op_def,sh_mem_store_def,
sh_mem_load_def] >>
metis_tac[PAIR,FST,SND])>>
gvs[Once evaluate_def,AllCaseEqs(),ELIM_UNCURRY,empty_locals_def,dec_clock_def,set_var_def] >>
metis_tac[PAIR,FST,SND])
QED
Definition every_exp_def:
(every_exp P (panLang$Const w) = P(Const w)) ∧
(every_exp P (Var v) = P(Var v)) ∧
(every_exp P (Label f) = P(Label f)) ∧
(every_exp P (Struct es) = (P(Struct es) ∧ EVERY (every_exp P) es)) ∧
(every_exp P (Field i e) = (P(Field i e) ∧ every_exp P e)) ∧
(every_exp P (Load sh e) = (P(Load sh e) ∧ every_exp P e)) ∧
(every_exp P (LoadByte e) = (P(LoadByte e) ∧ every_exp P e)) ∧
(every_exp P (Op bop es) = (P(Op bop es) ∧ EVERY (every_exp P) es)) ∧
(every_exp P (Panop op es) = (P(Panop op es) ∧ EVERY (every_exp P) es)) ∧
(every_exp P (Cmp c e1 e2) = (P(Cmp c e1 e2) ∧ every_exp P e1 ∧ every_exp P e2)) ∧
(every_exp P (Shift sh e num) = (P(Shift sh e num) ∧ every_exp P e)) ∧
(every_exp P BaseAddr = P BaseAddr) ∧
(every_exp P BytesInWord = P BytesInWord)
Termination
wf_rel_tac `measure (exp_size ARB o SND)` >>
rpt strip_tac >>
imp_res_tac MEM_IMP_exp_size >>
TRY (first_x_assum (assume_tac o Q.SPEC `ARB`)) >>
decide_tac
End
Definition exps_of_def:
(exps_of (Raise _ e) = [e]) ∧
(exps_of (Dec _ e p) = e::exps_of p) ∧
(exps_of (Seq p q) = exps_of p ++ exps_of q) ∧
(exps_of (If e p q) = e::exps_of p ++ exps_of q) ∧
(exps_of (While e p) = e::exps_of p) ∧
(exps_of (Call NONE e es) = e::es) ∧
(exps_of (Call (SOME (_ , (SOME (_ , _ , ep)))) e es) = e::es++exps_of ep) ∧
(exps_of (Call (SOME (_ , NONE)) e es) = e::es) ∧
(exps_of (DecCall _ _ e es p) = e::es++exps_of p) ∧
(exps_of (Store e1 e2) = [e1;e2]) ∧
(exps_of (StoreByte e1 e2) = [e1;e2]) ∧
(exps_of (Return e) = [e]) ∧
(exps_of (ExtCall _ e1 e2 e3 e4) = [e1;e2;e3;e4]) ∧
(exps_of (Assign _ e) = [e]) ∧
(exps_of (ShMemLoad _ _ e) = [e]) ∧
(exps_of (ShMemStore _ e1 e2) = [e1;e2]) ∧
(exps_of _ = [])
End
val _ = export_theory();