-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathLocusSem.v
executable file
·765 lines (675 loc) · 37.2 KB
/
LocusSem.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
Require Import Reals.
Require Import Psatz.
Require Import Complex.
Require Import SQIR.
Require Import VectorStates UnitaryOps Coq.btauto.Btauto Coq.NArith.Nnat Permutation.
Require Import Dirac.
Require Import QPE.
Require Import BasicUtility.
Require Import OQASM.
Require Import OQASMProof.
Require Import Classical_Prop.
Require Import MathSpec.
Require Import QafnySyntax.
Require Import LocusDef.
Require Import LocusKind.
(**********************)
(** Unitary Programs **)
(**********************)
Require Import Coq.FSets.FMapList.
Require Import Coq.FSets.FMapFacts.
Require Import Coq.Structures.OrderedTypeEx.
Declare Scope pexp_scope.
Delimit Scope pexp_scope with pexp.
Local Open Scope pexp_scope.
Local Open Scope nat_scope.
(* This is the semantics for basic gate set of the language. *)
Fixpoint compile_range_state (n st i:nat) (x:var) (b: rz_val) (f:posi -> val) :=
match n with 0 => f
| S m => (compile_range_state m st i x b f)[(x,st+m) |-> (nval (b (i+m)) allfalse)]
end.
Fixpoint compile_ses_state' (i:nat) (l:locus) (b:rz_val) :=
match l with nil => (fun _ => nval false allfalse)
| ((x,BNum l,BNum r)::xl) => compile_range_state (r-l) l i x b (compile_ses_state' (i+(r-l)) xl b)
| (_::xl) => compile_ses_state' i xl b
end.
Definition compile_ses_state (l:locus) (b:rz_val) := compile_ses_state' 0 l b.
Fixpoint turn_oqasm_range (rmax n st i:nat) (x:var) (f:posi -> val) (r:rz_val) (b: rz_val) : option (rz_val * rz_val) :=
match n with 0 => Some (r,b)
| S m => match (turn_oqasm_range rmax m st i x f r b)
with None => None
| Some (r',b') => match f (x,st+m) with nval ba ra => Some (n_rotate rmax ra r', update b' (i+m) ba)
| _ => None
end
end
end.
Fixpoint turn_oqasm_ses' (rmax i:nat) (l:locus) (f:posi -> val) (b:rz_val) :=
match l with nil => Some (allfalse, b)
| ((x,BNum l,BNum r)::xl) =>
match turn_oqasm_ses' rmax (i+(r-l)) xl f b with None => None
| Some (ra,ba) => turn_oqasm_range rmax (r-l) l i x f ra ba
end
| _ => None
end.
Definition turn_oqasm_ses rmax (l:locus) (f:posi -> val) b := turn_oqasm_ses' rmax 0 l f b.
Definition cover_n (f:rz_val) (n:nat) := fun i => if i <? n then false else f i.
Inductive match_value : nat -> state_elem -> state_elem -> Prop :=
match_nval : forall n p r1 r2, (forall i, i < n -> r1 i = r2 i) -> match_value n (Nval p r1) (Nval p r2)
| match_hval: forall n r1 r2, (forall i, i < n -> r1 i = r2 i) -> match_value n (Hval r1) (Hval r2)
| match_cval: forall n m r1 r2, (forall j, j < m -> fst (r1 j) = fst (r2 j) /\
(forall i, i < n -> (snd (r1 j)) i = (snd (r2 j)) i)) -> match_value n (Cval m r1) (Cval m r2).
Lemma match_value_trans: forall s1 s2 s3 n, match_value n s1 s2 -> match_value n s2 s3 -> match_value n s1 s3.
Proof.
intros. induction H; intros; simpl in *; try easy. inv H0.
constructor. intros. rewrite H; try lia. rewrite H5; try lia. easy.
inv H0.
constructor. intros. rewrite H; try lia. rewrite H3; try lia. easy.
inv H0.
constructor. intros. apply H in H0 as X1. apply H5 in H0 as X2.
destruct X1. destruct X2. split. rewrite H1. rewrite H3. easy.
intros. rewrite H2; try lia. rewrite H4; try lia. easy.
Qed.
Definition match_values (S1 S2: qstate) :=
Forall2 (fun s1 s2 => fst s1 = fst s2 /\
(match ses_len (fst s1) with Some n => match_value n (snd s1) (snd s2)
| None => False
end)) S1 S2.
Definition eval_nor (rmax:nat) (env:aenv) (l:locus) (r:C) (b:rz_val) (e:exp) :=
match compile_ses_qenv env l with (f,ss) =>
match compile_exp_to_oqasm e with
None => None
| Some e' =>
match ses_len l with None => None
| Some na =>
match turn_oqasm_ses rmax l (exp_sem f e' (compile_ses_state l b)) (cover_n b na)
with None => None
| Some (ra,ba) => Some ((r* (Cexp (2*PI * (turn_angle ra rmax))))%C, ba)
end
end
end
end.
(* The theorems were proved in VQO: https://github.com/inQWIRE/VQO *)
Definition turn_pair_nor (a:C *rz_val) := Nval (fst a) (snd a).
Axiom eval_nor_switch_same : forall rmax env l l1 n r b b1 e v, ses_len (l++l1) = Some n
-> (forall i, i < n -> b i = b1 i) ->
eval_nor rmax env l r b e = Some v ->
exists v', (eval_nor rmax env l r b1 e = Some v'
/\ match_value n (turn_pair_nor v) (turn_pair_nor v')).
Axiom compile_exp_WF : forall e ea l aenv qenv vl x v n, compile_ses_qenv aenv l = (qenv, vl)
-> compile_exp_to_oqasm e = Some ea -> type_exp aenv e (QT n, l) -> v >= qenv x
-> exp_WF qenv ea.
Axiom compile_exp_fresh : forall e ea l aenv qenv vl x v n, compile_ses_qenv aenv l = (qenv, vl)
-> compile_exp_to_oqasm e = Some ea -> type_exp aenv e (QT n, l) -> v >= qenv x
-> exp_fresh qenv (x,v) ea.
Fixpoint eval_ch (rmax:nat) (env:aenv) (l:locus) (m:nat) f (e:exp) :=
match m with 0 => Some (fun _ => (C0 , allfalse))
| S n => match eval_nor rmax env l (fst (f n)) (snd (f n)) e with None => None
| Some (ra,ba) => match eval_ch rmax env l n f e with None => None
| Some fa => Some (update fa n (ra , ba))
end
end
end.
(*The proof was given in VQO: https://github.com/inQWIRE/VQO*)
Lemma eval_ch_switch_same : forall m rmax env l l1 n b b1 e v, ses_len (l++l1) = Some n
-> (forall j : nat,
j < m ->
fst (b j) = fst (b1 j) /\
(forall i : nat, i < n -> snd (b j) i = snd (b1 j) i)) ->
eval_ch rmax env l m b e = Some v ->
(exists v', eval_ch rmax env l m b1 e = Some v' /\ match_value n (Cval m v) (Cval m v')).
Proof.
induction m; intros; simpl in *.
exists (fun _ : nat => (C0, allfalse)).
split; try easy. constructor. intros. lia.
destruct (eval_nor rmax env l (@fst C rz_val (b m))
(@snd C rz_val (b m)) e) eqn:eq1; try easy.
destruct p.
assert (A1 := H0).
specialize (H0 m). assert (m < S m) by lia.
assert ((@fst C (forall _ : nat, bool) (b m)) = (@fst C rz_val (b m))) by easy.
assert ((@snd C (forall _ : nat, bool) (b m)) = (@snd C rz_val (b m))) by easy.
apply H0 in H2. destruct H2. rewrite H3 in *. rewrite H2 in eq1.
apply eval_nor_switch_same with (l1 := l1)
(n := n) (b1 := snd (b1 m)) in eq1 as X1; try easy.
destruct X1 as [va [X1 X2]].
assert ((@fst C (forall _ : nat, bool) (b1 m)) = (@fst C rz_val (b1 m))) by easy.
assert ((@snd C (forall _ : nat, bool) (b1 m)) = (@snd C rz_val (b1 m))) by easy.
rewrite H6 in *. rewrite H7 in *.
rewrite X1. destruct va. simpl in *.
destruct (eval_ch rmax env l m b e) eqn:eq2; try easy.
apply IHm with (l1 := l1) (n := n) (b1 := b1) in eq2; try easy.
destruct eq2 as [va [Y1 Y2]]. rewrite Y1. inv H1.
exists (update va m (c0, r0)). split; try easy.
constructor. intros. split.
bdestruct (m =? j); subst.
repeat rewrite update_index_eq. inv X2. simpl. easy.
repeat rewrite update_index_neq; try lia. inv X2.
inv Y2. specialize (H12 j). assert (j < m) by lia.
apply H12 in H9. destruct H9. easy.
intros.
bdestruct (m =? j); subst.
repeat rewrite update_index_eq. inv X2. simpl. rewrite H11. easy. easy.
repeat rewrite update_index_neq; try lia. inv X2.
inv Y2. specialize (H12 j). assert (j < m) by lia.
apply H13 in H10. destruct H10. rewrite H11. easy. easy.
intros. apply A1. lia.
Qed.
Definition eval_to_had (n:nat) (b:rz_val) := (fun i => if i <? n then (update allfalse 0 (b i)) else allfalse).
Definition eval_to_nor (n:nat) (b:nat -> rz_val) := (fun i => if i <? n then b i 0 else false).
Lemma type_exp_exists_oqasm: forall env e n l, type_exp env e (QT n,l)
-> (exists e', compile_exp_to_oqasm e = Some e').
Proof.
intros. induction H; simpl in *; try repeat eauto.
destruct IHtype_exp. rewrite H3. exists (OQASM.CU (x,v) x0). auto.
destruct IHtype_exp1, IHtype_exp2. rewrite H3. rewrite H2.
exists (x; x0)%exp. auto.
Qed.
Definition all_nor_mode (f:posi -> val) := forall x n, right_mode_val OQASM.Nor (f (x,n)).
Lemma compile_range_state_nor: forall n s i x b f, all_nor_mode f
-> all_nor_mode (compile_range_state n s i x b f).
Proof.
intros. induction n;simpl in *. easy.
unfold all_nor_mode in *. intros.
bdestruct (posi_eq (x, s + n) (x0, n0)).
inv H0. rewrite eupdate_index_eq. constructor.
rewrite eupdate_index_neq; try easy.
Qed.
Lemma compile_ses_state'_nor: forall l i b,
all_nor_mode (compile_ses_state' i l b).
Proof.
induction l;intros;simpl in *.
unfold all_nor_mode. intros. constructor.
destruct a. destruct p. destruct b1. easy. destruct b0. easy.
apply compile_range_state_nor.
apply IHl.
Qed.
Lemma turn_oqasm_range_exists: forall n rmax st i x f r b, all_nor_mode f ->
exists v t, turn_oqasm_range rmax n st i x f r b = Some (v,t).
Proof.
intros.
induction n;simpl in *.
exists r,b. easy.
destruct IHn as [v [t X1]]. rewrite X1.
destruct (f (x, st + n)) eqn:eq1.
exists (n_rotate rmax r0 v), (update t (i + n) b0). easy.
unfold all_nor_mode in *.
specialize (H x (st+n)). inv H. rewrite eq1 in H0. inv H0.
Qed.
Lemma get_core_simple: forall l, simple_ses l -> (exists na, get_core_ses l = Some na).
Proof.
intros. induction l;try easy.
simpl in *. exists nil. easy.
simpl in *. inv H. unfold simple_bound in *. destruct x. easy. destruct y. easy.
apply IHl in H4. destruct H4. rewrite H. exists ((a0, n, n0) :: x).
easy.
Qed.
Lemma ses_len_simple: forall l, simple_ses l -> (exists na, ses_len l = Some na).
Proof.
intros. unfold ses_len. apply get_core_simple in H.
destruct H. rewrite H. exists (ses_len_aux x). easy.
Qed.
Lemma ses_simple_shrink: forall l l1, simple_ses (l ++ l1) -> simple_ses l.
Proof.
intros. induction l; intros; simpl in *; try easy. constructor.
inv H. constructor; try easy. apply IHl. easy.
Qed.
Lemma turn_oqasm_ses_simple: forall l n r f b, simple_ses l
-> all_nor_mode f -> exists na, turn_oqasm_ses' r n l f b = Some na.
Proof.
induction l; intros;simpl in *.
exists (allfalse, b); try easy.
destruct a. destruct p. inv H. unfold simple_bound in *.
destruct b1. easy. destruct b0; try easy.
apply (IHl (n + (n1 - n0)) r f b) in H7.
destruct H7. rewrite H. destruct x.
apply (turn_oqasm_range_exists (n1-n0) r n0 n v f b0 r0) in H0 as X1.
destruct X1 as [v0 [t0 X1]]. exists (v0,t0). easy. easy.
Qed.
Lemma right_mode_env_compile_ses : forall l aenv b n t, compile_ses_qenv aenv l = (n, t)
-> all_nor_mode (compile_ses_state l b) -> (forall x : Env.key, Env.MapsTo x OQASM.Nor (form_oenv t))
-> right_mode_env n (form_oenv t) (compile_ses_state l b).
Proof.
induction l;intros;simpl in *.
unfold right_mode_env in *. intros.
unfold all_nor_mode in *. destruct p. specialize (H0 v n0).
simpl in *. specialize (H1 v).
apply mapsto_always_same with (v1 := t0) in H1; subst; try easy.
simpl in *.
destruct a. destruct p.
destruct (AEnv.find (elt:=ktype) v aenv) eqn:eq1. destruct k.
unfold right_mode_env. intros. destruct p.
unfold all_nor_mode in *. specialize (H0 v0 n0). simpl in *.
specialize (H1 v0).
apply mapsto_always_same with (v1 := t0) in H1; subst; try easy.
destruct (compile_ses_qenv aenv l) eqn:eq2.
destruct (var_in_list l0 v) eqn:eq3. inv H.
unfold right_mode_env. intros. destruct p.
unfold all_nor_mode in *. specialize (H0 v0 n1). simpl in *.
specialize (H1 v0).
apply mapsto_always_same with (v1 := t0) in H1; subst; try easy.
unfold right_mode_env. intros. destruct p.
unfold all_nor_mode in *. specialize (H0 v0 n2). simpl in *.
specialize (H1 v0).
apply mapsto_always_same with (v1 := t0) in H1; subst; try easy.
unfold right_mode_env. intros. destruct p.
unfold all_nor_mode in *. specialize (H0 v0 n0). simpl in *.
specialize (H1 v0).
apply mapsto_always_same with (v1 := t0) in H1; subst; try easy.
Qed.
Lemma eval_nor_exists {rmax : nat} : forall aenv l n c b e,
type_exp aenv e (QT n,l) -> simple_ses l -> oracle_prop aenv l e
-> exists ba, eval_nor rmax aenv l c b e = Some ba.
Proof.
intros.
apply type_exp_exists_oqasm in H as X1.
destruct X1 as [ea X1].
apply ses_len_simple in H0 as X2. destruct X2.
unfold eval_nor. destruct (compile_ses_qenv aenv l) eqn:eq1. rewrite X1.
rewrite H2.
specialize (turn_oqasm_ses_simple l 0 rmax (exp_sem n0 ea (compile_ses_state l b)) (cover_n b x) H0) as X2.
assert (all_nor_mode (compile_ses_state l b)) as X3.
unfold compile_ses_state. apply compile_ses_state'_nor.
assert (right_mode_env n0 (form_oenv l0) (compile_ses_state l b)) as X4.
apply right_mode_env_compile_ses with (aenv := aenv); try easy.
unfold oracle_prop in *. rewrite eq1 in H1. rewrite X1 in H1.
destruct H1; try easy.
unfold oracle_prop in *.
rewrite eq1 in H1.
rewrite X1 in H1.
destruct H1 as [X5 X6].
apply well_typed_right_mode_pexp with (e := ea) (tenv' := (form_oenv l0)) in X4 as eq2; try easy.
assert (all_nor_mode (exp_sem n0 ea (compile_ses_state l b))).
unfold all_nor_mode, right_mode_env in *. intros.
specialize (X3 x0 n1). inv X3.
specialize (eq2 OQASM.Nor (x0,n1)). simpl in *.
bdestruct (n1 <? n0 x0). apply eq2 in H3; try easy.
apply (compile_exp_WF e ea l aenv n0 l0 x0 n1 n) in H3 as X7; try easy.
apply (compile_exp_fresh e ea l aenv n0 l0 x0 n1 n) in H3 as X8; try easy.
rewrite efresh_exp_sem_irrelevant; try easy.
assert (all_nor_mode (compile_ses_state l b)).
apply compile_ses_state'_nor. unfold all_nor_mode in *. apply H4.
apply (turn_oqasm_ses_simple l 0 rmax (exp_sem n0 ea (compile_ses_state l b)) (cover_n b x)) in H1 as X7.
destruct X7. unfold turn_oqasm_ses. rewrite H3. destruct x0.
exists ((c * Cexp (2 * PI * turn_angle b0 rmax))%C, r). easy. easy.
Qed.
Lemma eval_ch_exists {rmax : nat} : forall m aenv l n f e,
type_exp aenv e (QT n,l) -> simple_ses l -> oracle_prop aenv l e
-> exists ba, eval_ch rmax aenv l m f e = Some ba.
Proof.
induction m; intros;simpl in *.
exists (fun _ : nat => (C0, allfalse)).
easy.
apply (@eval_nor_exists rmax aenv l n (fst (f m)) (snd (f m)) e) in H as X1; try easy.
destruct X1. rewrite H2. destruct x.
apply (IHm aenv l n f) in H as X1; try easy. destruct X1.
rewrite H3.
exists (update x m (c, r)). easy.
Qed.
(* functions for defining boolean. *)
Fixpoint eval_eq_bool (f:nat -> C * rz_val) (m size v:nat) :=
match m with 0 => f
| S m' => update (eval_eq_bool f m' size v) m'
(fst (f m'),update (snd (f m')) size (xorb ((a_nat2fb (snd (f m')) size) =? v) ((snd (f m')) size)))
end.
Fixpoint eval_lt_bool (f:nat -> C * rz_val) (m size v:nat) :=
match m with 0 => f
| S m' => update (eval_lt_bool f m' size v) m'
(fst (f m'),update (snd (f m')) size (xorb ((a_nat2fb (snd (f m')) size) <? v) ((snd (f m')) size)))
end.
Fixpoint eval_rlt_bool (f:nat -> C * rz_val) (m size v:nat) :=
match m with 0 => f
| S m' => update (eval_rlt_bool f m' size v) m'
(fst (f m'),update (snd (f m')) size (xorb (v <? (a_nat2fb (snd (f m')) size)) ((snd (f m')) size)))
end.
Fixpoint grab_bool_elem (f:nat -> C * rz_val) (m size:nat) :=
match m with 0 => (0,(fun _ => (C0,allfalse)))
| S m' => if (snd (f m')) size then
match grab_bool_elem f m' size with (i,acc) => (i+1,update acc i (f m')) end
else grab_bool_elem f m' size
end.
Definition grab_bool f m size := grab_bool_elem f m (size - 1).
Axiom grab_bool_gt : forall f m size, m > 0 -> size > 0 -> fst (grab_bool f m size) > 0.
Inductive eval_bexp : qstate -> bexp -> qstate -> Prop :=
| beq_sem_1 : forall s x y z i l n m f,
eval_bexp (((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l,Cval m f)::s)
(BEq (BA x) ((Num y)) z (Num i))
(((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l,Cval m (eval_eq_bool f m n y))::s)
| beq_sem_2 : forall s x y z i l n m f,
eval_bexp (((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l,Cval m f)::s)
(BEq ((Num y)) (BA x) z (Num i))
(((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l,Cval m (eval_eq_bool f m n y))::s)
| blt_sem_1 : forall s x y z i l n m f,
eval_bexp ((((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l),Cval m f)::s)
(BLt (BA x) ((Num y)) z (Num i))
((((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l),(Cval m (eval_lt_bool f m n y)))::s)
| blt_sem_2 : forall s x y z i l n m f,
eval_bexp ((((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l),Cval m f)::s)
(BLt ((Num y)) (BA x) z (Num i))
((((x,BNum 0,BNum n)::(z,BNum i,BNum (S i))::l),(Cval m (eval_rlt_bool f m n y)))::s)
| btext_sem : forall s z i, eval_bexp s (BTest z (Num i)) s.
Lemma eval_bexp_det: forall q b q1 q2, eval_bexp q b q1 -> eval_bexp q b q2 -> q1 = q2.
Proof.
intros. induction b;intros;simpl in *.
inv H. inv H. inv H0. easy. inv H0. easy.
inv H. inv H0. easy. inv H0. easy. inv H. inv H0. easy.
inv H.
Qed.
Inductive find_basis_elems (n n':nat) (f:rz_val) (fc:nat -> C*rz_val):
nat -> nat -> (nat -> C * rz_val) -> Prop :=
find_basis_empty: find_basis_elems n n' f fc 0 0 (fun _ => (C0,allfalse))
| find_basis_many_1: forall i m acc, find_basis_elems n n' f fc i m acc ->
f = cut_n (lshift_fun (snd (fc i)) n') n
-> find_basis_elems n n' f fc (S i) (S m) (update acc m (fc i))
| find_basis_many_2: forall i m acc, find_basis_elems n n' f fc i m acc ->
f <> cut_n (lshift_fun (snd (fc i)) n') n -> find_basis_elems n n' f fc (S i) m acc.
(* The proof has been given in VQO based on the fact of quantum states. *)
Axiom find_basis_elems_same: forall m1 n n1 f r r' mv fv,
(forall j : nat,
j < m1 ->
fst (r j) = fst (r' j) /\
(forall i : nat, i < n1 -> snd (r j) i = snd (r' j) i)) ->
find_basis_elems n n1 f r m1 mv fv ->
(exists fv', find_basis_elems n n1 f r' m1 mv fv' /\ match_value n1 (Cval mv fv) (Cval mv fv')).
Inductive assem_elem : nat -> nat -> rz_val -> (nat-> C * rz_val) -> list nat -> Prop :=
assem_elem_0 : forall size c f, assem_elem 0 size c f nil
| assem_elem_st : forall size c f m l, assem_elem m size c f l -> cut_n (snd (f m)) size = c
-> assem_elem (S m) size c f (m::l)
| assem_elem_sf : forall size c f m l, assem_elem m size c f l -> cut_n (snd (f m)) size <> c
-> assem_elem (S m) size c f l.
Definition combine_two (n:nat) (f:rz_val) (g:rz_val) :=
(fun x => if x <? n then f x else g (x-n)).
Fixpoint assem_list (m base n:nat) (f:rz_val) (fc: nat -> C * rz_val) (acc:nat -> C*rz_val) :=
match m with 0 => (base, acc)
| S m' => match assem_list m' base n f fc acc with (mv, fv) =>
(S mv, (update fv mv (fst (fc m'), combine_two n f (snd (fc m')))))
end
end.
Lemma assem_list_num_eq: forall m base n f fc acc mv fv,
assem_list m base n f fc acc = (mv, fv) -> mv = base + m.
Proof.
induction m; intros; simpl in *; try easy. inv H. lia.
destruct (assem_list m base n f fc acc) eqn:eq1.
apply IHm in eq1; subst. inv H. lia.
Qed.
Lemma assem_list_same: forall m base n n1 bl r r' f f' mv fv,
(forall j : nat,
j < m ->
fst (r j) = fst (r' j) /\
(forall i : nat, i < n1 -> snd (r j) i = snd (r' j) i)) ->
(forall j : nat,
j < base ->
fst (f j) = fst (f' j) /\
(forall i : nat, i < n + n1 -> snd (f j) i = snd (f' j) i)) ->
assem_list m base n bl r f = (mv, fv) ->
(exists fv', assem_list m base n bl r' f' = (mv,fv')
/\ match_value (n + n1) (Cval mv fv) (Cval mv fv')).
Proof.
induction m; intros; simpl in *; try easy. inv H1.
exists f'. split; try easy.
constructor. intros. apply H0. easy.
destruct (assem_list m base n bl r f) eqn:eq1.
apply IHm with (n1 := n1) (r' := r') (f' := f') in eq1; try easy.
inv H1. destruct eq1 as [fv' [X1 X2]].
inv X2. rewrite X1; simpl in *.
exists (update fv' n0 (fst (r' m), combine_two n bl (snd (r' m)))).
split; try easy.
constructor. intros.
apply assem_list_num_eq in X1 as Y1; subst.
bdestruct (j =? base + m); subst.
repeat rewrite update_index_eq.
assert (m < S m) by lia.
apply H in H2. simpl in *. destruct H2.
replace ((@fst C (forall _ : nat, bool) (r m)))
with (@fst C rz_val (r m)) in * by easy.
rewrite H2. split; try easy.
intros. unfold combine_two in *.
bdestruct (i <? n). easy.
rewrite H4. easy. lia.
repeat rewrite update_index_neq by lia.
apply H3. lia.
intros. apply H. lia.
Qed.
(* first n is length of l and second is length of l1. third is num of elements *)
Inductive assem_bool (n n':nat): nat -> (nat-> C * rz_val) -> state_elem -> state_elem -> Prop :=
assem_bool_empty: forall f fc, assem_bool n n' 0 f fc (Cval 0 (fun _ => (C0,allfalse)))
| assem_bool_many_1: forall i m m' f fc acc fv, assem_bool n n' i f (Cval m fc) (Cval m' acc) ->
find_basis_elems n n' (cut_n (snd (f i)) n) fc m 0 fv ->
assem_bool n n' (S i) f (Cval m fc) (Cval (S m') (update acc m' (f i)))
| assem_bool_many_2: forall i m m' f fc acc mv fv ma fa, assem_bool n n' i f (Cval m fc) (Cval m' acc) ->
0 < mv -> find_basis_elems n n' (cut_n (snd (f i)) n) fc m mv fv ->
assem_list mv m' n (cut_n (snd (f i)) n) fv acc = (ma, fa) ->
assem_bool n n' (S i) f (Cval m fc) (Cval ma fa).
Lemma assem_bool_gt : forall i n n' f m fc ma fa,
assem_bool n n' i f (Cval m fc) (Cval ma fa) -> ma >= i.
Proof.
induction i; intros; simpl in *; try easy.
lia. inv H. apply IHi in H5. lia.
apply IHi in H6. apply assem_list_num_eq in H9; subst. lia.
Qed.
Lemma assem_bool_cval: forall n n' m f mv fv fca,
assem_bool n n' m f (Cval mv fv) fca -> (exists mv' fv', fca = Cval mv' fv').
Proof.
induction m; intros; simpl in *. inv H.
exists 0,(fun _ : nat => (C0, allfalse)). easy.
inv H. exists (S m'), (update acc m' (f m)). easy.
exists ma,fa. easy.
Qed.
Fixpoint subst_qstate (l:qstate) (x:var) (n:nat) :=
match l with nil => nil
| (y,v)::yl => (subst_locus y x n,v)::(subst_qstate yl x n)
end.
Definition subst_state (l:state) (x:var) n := (fst l,subst_qstate (snd l) x n).
(* defining diffusion. *)
Fixpoint sum_c_list (l:list nat) (f: nat -> C * rz_val) :=
match l with nil => C0
| a::xl => (fst (f a) + sum_c_list xl f)%C
end.
Definition cal_fc (l:list nat) (f:nat -> C*rz_val) (n:nat) :=
((1/(2^(n-1)))%R * (sum_c_list l f))%C.
Fixpoint trans_nat (l:list nat) (f:nat -> C*rz_val) (size:nat) :=
match l with nil => nil
| a::xl => (a_nat2fb (snd (f a)) size, fst (f a)):: trans_nat xl f size
end.
Fixpoint find_c (l:list (nat * C)) (a:nat) :=
match l with nil => None
| ((x,y)::xl) => if x =? a then Some y else find_c xl a
end.
Definition merge_two (n:nat) (c:rz_val) (size:nat) :=
fun i => if i <? size then (nat2fb n) i else c i.
Fixpoint assem_dis (l:list (nat * C)) (c:rz_val) (f:nat -> C*rz_val) (sum:C) (size:nat) (i:nat) (n:nat) :=
match n with 0 => f
| S m => match find_c l m with None
=> assem_dis l c (update f i (sum,merge_two m c size)) sum size (i+1) m
| Some z => assem_dis l c (update f i ((sum - z)%C,merge_two m c size)) sum size (i+1) m
end
end.
Inductive find_same_group : nat -> nat -> nat -> rz_val -> (nat -> C * rz_val) -> list nat -> Prop :=
find_same_group_0 : forall pos size c f, find_same_group 0 pos size c f nil
| find_same_group_st: forall n pos size c f l, find_same_group n pos size c f l ->
cut_n (lshift_fun (snd (f n)) pos) size = c -> find_same_group (S n) pos size c f (n::l)
| find_same_group_sf: forall n pos size c f l, find_same_group n pos size c f l ->
cut_n (lshift_fun (snd (f n)) pos) size <> c -> find_same_group (S n) pos size c f l.
Inductive dis_sem : nat -> nat -> nat -> nat -> (nat -> C * rz_val) -> (list nat) -> (nat * (nat -> C * rz_val)) -> Prop :=
dis_sem_0 : forall pos size n f l, dis_sem pos size n 0 f l (0,fun _ => (C0,allfalse))
| dis_sem_sf: forall pos size n m f l acc, dis_sem pos size n m f l acc -> In m l -> dis_sem pos size n (S m) f l acc
| dis_sem_st: forall pos size n m f l acc la, dis_sem pos size n m f (la++l) acc -> ~ In m l ->
find_same_group pos size n (cut_n (lshift_fun (snd (f m)) pos) size) f la ->
dis_sem pos size n (S m) f l
(fst acc+2^pos,assem_dis (trans_nat la f pos) (snd (f m)) (snd acc) (sum_c_list la f) pos (fst acc) (2^pos)).
(*big step semantics. *)
Inductive ForallA (rmax:nat) (P: nat -> aenv -> state -> pexp -> state -> Prop):
nat -> aenv -> state -> nat -> var -> bexp -> pexp -> state -> Prop :=
| ForallA_nil : forall env l x b e s, ForallA rmax P 0 env s l x b e s
| ForallA_cons : forall n env l x b e s s' s'', ForallA rmax P n env s l x b e s' ->
P rmax env s' (If (subst_bexp b x (l+n)) (subst_pexp e x (l+n))) s''
-> ForallA rmax P (S n) env s l x b e s''.
Inductive qfor_sem {rmax:nat}
: aenv -> state -> pexp -> state -> Prop :=
(*
| state_eq_sem: forall aenv e s f f' S, @state_equiv rmax f f' -> qfor_sem aenv (s,f') e S -> qfor_sem aenv (s,f) e S
*)
| skip_sem : forall aenv s, qfor_sem aenv s PSKIP s
| let_sem_c : forall aenv s s' x a n e, simp_aexp a = Some n -> qfor_sem aenv s (subst_pexp e x n) s'
-> qfor_sem aenv s (Let x (AE a) e) s'
| let_sem_m : forall aenv s s' W x a n e, eval_aexp (fst s) a n
-> qfor_sem (AEnv.add x (Mo MT) aenv) (update_cval s x n) e (W,s')
-> qfor_sem aenv s (Let x (AE a) e) (fst s,s')
| let_sem_q : forall aenv W W' s s' l x a n e r v va va', AEnv.MapsTo a (QT n) aenv ->
@pick_mea n va (r,v) -> build_state_ch n v va = Some va' ->
qfor_sem (AEnv.add x (Mo MT) aenv) (AEnv.add x (r,v) W, (l,va')::s) e (W',s')
-> qfor_sem aenv (W,((a,BNum 0,BNum n)::l,va)::s) (Let x (Meas a) e) (W,s')
| appu_sem_nor : forall aenv W s a e r b ra ba,
eval_nor rmax aenv a r b e = Some (ra,ba) -> qfor_sem aenv (W,(a,Nval r b)::s) (AppU a e) (W,(a,Nval ra ba)::s)
| appu_sem_ch : forall aenv W s a e l b m ba, eval_ch rmax aenv a m b e = Some ba
-> qfor_sem aenv (W,(a++l,Cval m b)::s) (AppU a e) (W,(a++l,Cval m ba)::s)
| appsu_sem_h_nor : forall aenv W s p a r b n, @simp_varia aenv p a -> ses_len ([a]) = Some n ->
qfor_sem aenv (W,([a],Nval r b)::s) (AppSU (RH p)) (W,([a],(Hval (eval_to_had n b)) )::s)
| appsu_sem_h_had : forall aenv W s p a b n, @simp_varia aenv p a -> ses_len ([a]) = Some n ->
qfor_sem aenv (W,([a],Hval b)::s) (AppSU (RH p)) (W,([a],(Nval C1 (eval_to_nor n b)))::s)
(* rewrite the tenv type for oqasm with respect to the ch list type. *)
| if_sem_ct : forall aenv s s' b e, simp_bexp b = Some true -> qfor_sem aenv s e s' -> qfor_sem aenv s (If b e) s'
| if_sem_cf : forall aenv s b e, simp_bexp b = Some false -> qfor_sem aenv s (If b e) s
(*
| if_sem_mt : forall aenv M s S b e, eval_cbexp M b true -> qfor_sem aenv (M,s) e S -> qfor_sem aenv (M,s) (If b e) S
| if_sem_mf : forall aenv M s b e, eval_cbexp M b false -> qfor_sem aenv (M,s) (If b e) (M,s)
*)
| if_sem_q : forall aenv W W' l l1 n n' s s' b e m f f' fc fc' fc'',
type_bexp aenv b (QT n,l) -> @eval_bexp ((l++l1, Cval m f)::s) b ((l++l1, Cval m f')::s) -> ses_len l1 = Some n' ->
mut_state 0 n n' (Cval (fst (grab_bool f' m n)) (snd (grab_bool f' m n))) fc ->
qfor_sem aenv (W,(l1,fc)::s) e (W',(l1,fc')::s') -> assem_bool n n' m f' fc' fc'' ->
qfor_sem aenv (W,(l++l1, Cval m f)::s) (If b e) (W',(l++l1, fc'')::s')
(* | diffuse_a: forall aenv s s' x n l n' l1 m f m' acc, type_vari aenv x (QT n,l) ->
@find_state rmax s l (Some (l++l1, Cval m f)) -> ses_len l1 = Some n' ->
dis_sem n n' m m f nil (m',acc) -> @up_state rmax s (l++l1) (Cval m' acc) s' ->
qfor_sem aenv s (Diffuse x) s' *)
| seq_sem: forall aenv e1 e2 s s1 s2, qfor_sem aenv s e1 s1 -> qfor_sem aenv s1 e2 s2 -> qfor_sem aenv s (PSeq e1 e2) s2
| for_sem: forall aenv s s' x l h b p, ForallA rmax (@qfor_sem) (h-l) aenv s l x b p s'
-> qfor_sem aenv s (For x (Num l) (Num h) b p) s'.
Definition qfor_sem_ind':
forall (rmax:nat) (P : nat -> aenv -> state -> pexp -> state -> Prop),
(*
(forall (aenv: aenv) (e:pexp) (W:stack) (s s':qstate) (S:state),
@state_equiv rmax s s' -> @qfor_sem rmax aenv (W,s') e S -> P rmax aenv (W,s') e S ->
P rmax aenv (W,s) e S) ->
*)
(forall (aenv: aenv) (S:state), P rmax aenv S PSKIP S) ->
(forall (aenv: aenv) (s s':state) (x:var) (a:aexp) (n:nat) (e:pexp),
simp_aexp a = Some n -> @qfor_sem rmax aenv s (subst_pexp e x n) s' -> P rmax aenv s (subst_pexp e x n) s' ->
P rmax aenv s (Let x (AE a) e) s') ->
(forall (aenv: aenv) (s :state) (s' : qstate) (W:stack) (x:var) (a:aexp) (n:R * nat) (e:pexp),
eval_aexp (fst s) a n -> @qfor_sem rmax (AEnv.add x (Mo MT) aenv) (update_cval s x n) e (W,s') ->
P rmax (AEnv.add x (Mo MT) aenv) (update_cval s x n) e (W,s') ->
P rmax aenv s (Let x (AE a) e) (fst s,s')) ->
(forall (aenv: aenv) (W W':stack) (s s' :qstate) (l:locus) (x:var) (a:var) (n:nat) (e:pexp) (r:R) (v:nat) (va va':state_elem),
AEnv.MapsTo a (QT n) aenv -> @pick_mea n va (r,v) -> build_state_ch n v va = Some va' ->
@qfor_sem rmax (AEnv.add x (Mo MT) aenv) (AEnv.add x (r,v) W, (l,va')::s) e (W',s') ->
P rmax (AEnv.add x (Mo MT) aenv) (AEnv.add x (r,v) W, (l,va')::s) e (W',s')
-> P rmax aenv (W,((a,BNum 0,BNum n)::l,va)::s) (Let x (Meas a) e) (W,s')) ->
(forall (aenv: aenv) (W:stack) (s:qstate) (a:locus) (e:exp) (r:C) (b : rz_val) (ra:C) (ba:rz_val),
eval_nor rmax aenv a r b e = Some (ra,ba) ->
P rmax aenv (W,(a,Nval r b)::s) (AppU a e) (W,(a,Nval ra ba)::s)) ->
(forall (aenv: aenv) (W:stack) (s:qstate) (a:locus) (e:exp) (l:locus) (b : nat -> C * rz_val) (m:nat) (ba:nat -> C * rz_val),
eval_ch rmax aenv a m b e = Some ba ->
P rmax aenv (W,(a++l,Cval m b)::s) (AppU a e) (W,(a++l,Cval m ba)::s)) ->
(forall (aenv: aenv) (W:stack) (s:qstate) (p:varia) (a:range) (r:C) (b : rz_val) (n:nat),
@simp_varia aenv p a -> ses_len ([a]) = Some n ->
P rmax aenv (W,([a],Nval r b)::s) (AppSU (RH p)) (W,([a],(Hval (eval_to_had n b)) )::s)) ->
(forall (aenv: aenv) (W:stack) (s:qstate) (p:varia) (a:range) (b : nat -> rz_val) (n:nat),
@simp_varia aenv p a -> ses_len ([a]) = Some n ->
P rmax aenv (W,([a],Hval b)::s) (AppSU (RH p)) (W,([a],(Nval C1 (eval_to_nor n b)))::s)) ->
(forall (aenv: aenv) (s s':state) (b:bexp) (e:pexp),
simp_bexp b = Some true -> @qfor_sem rmax aenv s e s' -> P rmax aenv s e s' -> P rmax aenv s (If b e) s') ->
(forall (aenv: aenv) (s:state) (b:bexp) (e:pexp),
simp_bexp b = Some false -> P rmax aenv s (If b e) s) ->
(forall (aenv : aenv) (W W' : stack) (l : locus)
(l1 : list range) (n n' : nat)
(s s' : list (list range * state_elem))
(b : bexp) (e : pexp) (m : nat) (f f' : nat -> C * rz_val)
(fc fc' fc'': state_elem),
type_bexp aenv b (QT n,l) -> @eval_bexp ((l++l1, Cval m f)::s) b ((l++l1, Cval m f')::s) -> ses_len l1 = Some n' ->
mut_state 0 n n' (Cval (fst (grab_bool f' m n)) (snd (grab_bool f' m n))) fc ->
@qfor_sem rmax aenv (W,(l1,fc)::s) e (W',(l1,fc')::s') ->
P rmax aenv (W,(l1,fc)::s) e (W',(l1,fc')::s') -> assem_bool n n' m f' fc' fc'' ->
P rmax aenv (W,(l++l1, Cval m f)::s) (If b e) (W',(l++l1, fc'')::s')) ->
(forall (aenv: aenv) (e1 e2:pexp) (s s1 s2:state),
@qfor_sem rmax aenv s e1 s1 -> P rmax aenv s e1 s1 ->
@qfor_sem rmax aenv s1 e2 s2 -> P rmax aenv s1 e2 s2 -> P rmax aenv s (PSeq e1 e2) s2) ->
(forall (aenv: aenv) (s s':state) (x:var) (l h:nat) (b:bexp) (p:pexp),
ForallA rmax (@qfor_sem) (h-l) aenv s l x b p s' -> ForallA rmax P (h-l) aenv s l x b p s' ->
P rmax aenv s (For x (Num l) (Num h) b p) s') ->
(forall (aenv: aenv) (S:state) (e:pexp) (S':state), @qfor_sem rmax aenv S e S' -> P rmax aenv S e S').
Proof.
intros rmax P ST2 ST3 ST4 ST5 ST6 ST7 ST8 ST9 ST10 ST11 ST12 ST13 ST14.
refine
(fix PF aenv S e S' (Hw : @qfor_sem rmax aenv S e S') {struct Hw} :=
match Hw with
(* | state_eq_sem aenv e W s s' Sa Hseq Hw => ST1 aenv e W s s' Sa Hseq Hw _ *)
| skip_sem aenv s => ST2 aenv s
| let_sem_c aenv s s' x a n e Hev Hw => ST3 aenv s s' x a n e Hev Hw _
| let_sem_m aenv s s' W x a n e Hev Hw => ST4 aenv s s' W x a n e Hev Hw _
| let_sem_q aenv W W' s s' l x a n e r v va va' Henv Hpick Hbd Hw =>
ST5 aenv W W' s s' l x a n e r v va va' Henv Hpick Hbd Hw _
| appu_sem_nor aenv W s a e r b ra ba Hev => ST6 aenv W s a e r b ra ba Hev
| appu_sem_ch aenv W s a e l b m ba Hev => ST7 aenv W s a e l b m ba Hev
| appsu_sem_h_nor aenv W s p a r b n Hsp Hlen => ST8 aenv W s p a r b n Hsp Hlen
| appsu_sem_h_had aenv W s p a b n Hsp Hlen => ST9 aenv W s p a b n Hsp Hlen
| if_sem_ct aenv s s' b e Hsp Hw => ST10 aenv s s' b e Hsp Hw _
| if_sem_cf aenv s b e Hsp => ST11 aenv s b e Hsp
| if_sem_q aenv W W' l l1 n n' s s' b e m f f' fc fc' fc'' Hty Hev Hses Hmut1 Hw Has
=> ST12 aenv W W' l l1 n n' s s' b e m f f' fc fc' fc'' Hty Hev Hses Hmut1 Hw _ Has
| seq_sem aenv e1 e2 s s1 s2 Hw1 Hw2 => ST13 aenv e1 e2 s s1 s2 Hw1 _ Hw2 _
| for_sem aenv s s' x l h b p Hw => ST14 aenv s s' x l h b p Hw _
end).
(* exact (PF aenv (W, s') e Sa Hw). *)
exact (PF aenv s (subst_pexp e x n) s' Hw).
exact (PF (AEnv.add x (Mo MT) aenv) (update_cval s x n) e (W,s') Hw).
exact (PF (AEnv.add x (Mo MT) aenv) (AEnv.add x (r, v) W, (l, va') :: s) e (W',s') Hw).
exact (PF aenv s e s' Hw).
exact (PF aenv (W, (l1, fc) :: s) e (W', (l1, fc') :: s') Hw).
exact (PF aenv s e1 s1 Hw1).
exact (PF aenv s1 e2 s2 Hw2).
remember (h-l) as n. clear Heqn.
induction Hw. constructor.
apply PF in H. apply ForallA_cons with (s' := s'). easy. easy.
Qed.
(*small step semantics. *)
Inductive step {rmax:nat}
: aenv -> state -> pexp -> state -> pexp -> Prop :=
| state_eq_step: forall aenv e e' s f f' S, @state_equiv rmax f f' -> step aenv (s,f') e S e' -> step aenv (s,f) e S e'
| let_step_c : forall aenv s x a n e, simp_aexp a = Some n
-> step aenv s (Let x (AE a) e) s (subst_pexp e x n)
| let_step_m : forall aenv s x a n e, eval_aexp (fst s) a n
-> step aenv s (Let x (AE a) e) (update_cval s x n) e
| let_step_q : forall aenv W s l x a n e r v va va', AEnv.MapsTo a (QT n) aenv ->
@pick_mea n va (r,v) -> build_state_ch n v va = Some va'
-> step (AEnv.add x (Mo MT) aenv) (W,((a,BNum 0,BNum n)::l,va)::s) (Let x (Meas a) e) (AEnv.add x (r,v) W, (l,va')::s) e
| appu_step_nor : forall aenv W s a e r b ra ba, eval_nor rmax aenv a r b e = Some (ra,ba)
-> step aenv (W,(a,Nval r b)::s) (AppU a e) (W,(a,Nval ra ba)::s) PSKIP
| appu_step_ch : forall aenv W s a e l b m ba, eval_ch rmax aenv a m b e = Some ba
-> step aenv (W,(a++l,Cval m b)::s) (AppU a e) (W,(a++l,Cval m ba)::s) PSKIP
| appsu_step_h_nor : forall aenv W s p a r b, @simp_varia aenv p a ->
step aenv (W,([a],Nval r b)::s) (AppSU (RH p)) (W,([a],(Hval (fun i => (update allfalse 0 (b i)))) )::s) PSKIP
| appsu_step_h_had : forall aenv W s p a b, @simp_varia aenv p a
-> step aenv (W,([a],Hval b)::s) (AppSU (RH p)) (W,([a],(Nval C1 (fun j => b j 0)))::s) PSKIP
(* rewrite the tenv type for oqasm with respect to the ch list type. *)
| seq_step_1: forall aenv e1 e1' e2 s s1 s2, e1 <> PSKIP -> step aenv s e1 s1 e1' -> step aenv s (PSeq e1 e2) s2 (PSeq e1' e2)
| seq_step_2: forall aenv e2 s, step aenv s (PSeq PSKIP e2) s e2
| if_step_ct : forall aenv s s' b e e', simp_bexp b = Some true -> step aenv s e s' e' -> step aenv s (If b e) s (If b e')
| if_step_cf : forall aenv s b e, simp_bexp b = Some false -> step aenv s (If b e) s PSKIP
| if_step_mt : forall aenv M s s' b e e', eval_cbexp M b true -> step aenv (M,s) (If b e) s' e' -> step aenv (M,s) (If b e) s' (If b e')
| if_step_mf : forall aenv M s b e, eval_cbexp M b false -> step aenv (M,s) (If b e) (M,s) PSKIP
| if_step_q : forall aenv s b, step aenv s (If b PSKIP) s PSKIP
| for_step_0 : forall aenv s x l h b p, h <= l -> step aenv s (For x (Num l) (Num h) b p) s PSKIP
| for_step_s : forall aenv s x l h b p, l < h ->
step aenv s (For x (Num l) (Num h) b p) s (PSeq (If b p) (For x (Num (l+1)) (Num h) b p))
(* | diffuse_step_a: forall aenv s s' x n l n' l1 m f m' acc, type_vari aenv x (QT n,l) ->
@find_state rmax s l (Some (l++l1, Cval m f)) -> ses_len l1 = Some n' ->
dis_sem n n' m m f nil (m',acc) -> @up_state rmax s (l++l1) (Cval m' acc) s' ->
step aenv s (Diffuse x) s' PSKIP
*).
(*
Inductive locus_system {rmax:nat}
: atype -> aenv -> type_map -> pexp -> type_map -> Prop :=
Inductive qfor_sem {rmax:nat}
: aenv -> state -> pexp -> state -> Prop :=
*)