-
Notifications
You must be signed in to change notification settings - Fork 12
/
CompactSemantics.v
182 lines (165 loc) · 7.69 KB
/
CompactSemantics.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
(*! Language | Semantics of typed Kôika programs with compact logs !*)
Require Export Koika.Common Koika.Environments Koika.CompactLogs Koika.Syntax Koika.TypedSyntax.
Section Interp.
Context {pos_t var_t fn_name_t rule_name_t reg_t ext_fn_t: Type}.
Context {reg_t_eq_dec: EqDec reg_t}.
Context {R: reg_t -> type}.
Context {Sigma: ext_fn_t -> ExternalSignature}.
Context {REnv: Env reg_t}.
Notation Log := (Log R REnv).
Notation rule := (rule pos_t var_t fn_name_t R Sigma).
Notation action := (action pos_t var_t fn_name_t R Sigma).
Notation scheduler := (scheduler pos_t rule_name_t).
Definition tcontext (sig: tsig var_t) :=
context (fun k_tau => type_denote (snd k_tau)) sig.
Definition acontext (sig argspec: tsig var_t) :=
context (fun k_tau => action sig (snd k_tau)) argspec.
Section Action.
Context (r: REnv.(env_t) R).
Context (sigma: forall f, Sig_denote (Sigma f)).
Section Args.
Context (interp_action:
forall {sig: tsig var_t} {tau}
(Gamma: tcontext sig)
(sched_log: Log) (action_log: Log)
(a: action sig tau),
option (Log * type_denote tau * (tcontext sig))).
Fixpoint interp_args'
{sig: tsig var_t}
(Gamma: tcontext sig)
(sched_log: Log)
(action_log: Log)
{argspec: tsig var_t}
(args: acontext sig argspec)
: option (Log * tcontext argspec * (tcontext sig)) :=
match args with
| CtxEmpty => Some (action_log, CtxEmpty, Gamma)
| @CtxCons _ _ argspec k_tau arg args =>
let/opt3 action_log, ctx, Gamma := interp_args' Gamma sched_log action_log args in
let/opt3 action_log, v, Gamma := interp_action _ _ Gamma sched_log action_log arg in
Some (action_log, CtxCons k_tau v ctx, Gamma)
end.
End Args.
Fixpoint interp_action
{sig: tsig var_t}
{tau}
(Gamma: tcontext sig)
(sched_log: Log)
(action_log: Log)
(a: action sig tau)
{struct a}
: option (Log * tau * (tcontext sig)) :=
match a in TypedSyntax.action _ _ _ _ _ ts tau return (tcontext ts -> option (Log * tau * (tcontext ts))) with
| Fail tau => fun _ =>
None
| Var m => fun Gamma =>
Some (action_log, cassoc m Gamma, Gamma)
| Const cst => fun Gamma =>
Some (action_log, cst, Gamma)
| Seq r1 r2 => fun Gamma =>
let/opt3 action_log, _, Gamma := interp_action Gamma sched_log action_log r1 in
interp_action Gamma sched_log action_log r2
| @Assign _ _ _ _ _ _ _ _ k tau m ex => fun Gamma =>
let/opt3 action_log, v, Gamma := interp_action Gamma sched_log action_log ex in
Some (action_log, Ob, creplace m v Gamma)
| @Bind _ _ _ _ _ _ _ sig tau tau' var ex body => fun (Gamma : tcontext sig) =>
let/opt3 action_log1, v, Gamma := interp_action Gamma sched_log action_log ex in
let/opt3 action_log2, v, Gamma := interp_action (CtxCons (var, tau) v Gamma) sched_log action_log1 body in
Some (action_log2, v, ctl Gamma)
| If cond tbranch fbranch => fun Gamma =>
let/opt3 action_log, cond, Gamma := interp_action Gamma sched_log action_log cond in
if Bits.single cond then
interp_action Gamma sched_log action_log tbranch
else
interp_action Gamma sched_log action_log fbranch
| Read P0 idx => fun Gamma =>
if may_read0 sched_log idx then
Some (Environments.update
REnv action_log idx
(fun rl => {| lread0 := true; lread1 := rl.(lread1);
lwrite0 := rl.(lwrite0); lwrite1 := rl.(lwrite1) |}),
REnv.(getenv) r idx,
Gamma)
else None
| Read P1 idx => fun Gamma =>
if may_read1 sched_log idx then
Some (Environments.update
REnv action_log idx
(fun rl => {| lread0 := rl.(lread1); lread1 := true;
lwrite0 := rl.(lwrite0); lwrite1 := rl.(lwrite1) |}),
match (REnv.(getenv) action_log idx).(lwrite0), (REnv.(getenv) sched_log idx).(lwrite0) with
| Some v, _ => v
| _, Some v => v
| _, _ => REnv.(getenv) r idx
end,
Gamma)
else None
| Write P0 idx val => fun Gamma =>
let/opt3 action_log, val, Gamma_new := interp_action Gamma sched_log action_log val in
if may_write0 sched_log action_log idx then
Some (Environments.update
REnv action_log idx
(fun rl => {| lread0 := rl.(lread1); lread1 := rl.(lread1);
lwrite0 := Some val; lwrite1 := rl.(lwrite1) |}),
Bits.nil, Gamma_new)
else None
| Write P1 idx val => fun Gamma =>
let/opt3 action_log, val, Gamma_new := interp_action Gamma sched_log action_log val in
if may_write1 sched_log action_log idx then
Some (Environments.update
REnv action_log idx
(fun rl => {| lread0 := rl.(lread1); lread1 := rl.(lread1);
lwrite0 := rl.(lwrite0); lwrite1 := Some val |}),
Bits.nil, Gamma_new)
else None
| Unop fn arg1 => fun Gamma =>
let/opt3 action_log, arg1, Gamma := interp_action Gamma sched_log action_log arg1 in
Some (action_log, (PrimSpecs.sigma1 fn) arg1, Gamma)
| Binop fn arg1 arg2 => fun Gamma =>
let/opt3 action_log, arg1, Gamma := interp_action Gamma sched_log action_log arg1 in
let/opt3 action_log, arg2, Gamma := interp_action Gamma sched_log action_log arg2 in
Some (action_log, (PrimSpecs.sigma2 fn) arg1 arg2, Gamma)
| ExternalCall fn arg1 => fun Gamma =>
let/opt3 action_log, arg1, Gamma := interp_action Gamma sched_log action_log arg1 in
Some (action_log, sigma fn arg1, Gamma)
| InternalCall name args body => fun Gamma =>
let/opt3 action_log, results, Gamma := interp_args' (@interp_action) Gamma sched_log action_log args in
let/opt3 action_log, v, _ := interp_action results sched_log action_log body in
Some (action_log, v, Gamma)
| APos _ a => fun Gamma =>
interp_action Gamma sched_log action_log a
end Gamma.
Definition interp_rule (sched_log: Log) (rl: rule) : option Log :=
match interp_action CtxEmpty sched_log log_empty rl with
| Some (l, _, _) => Some l
| None => None
end.
End Action.
Section Scheduler.
Context (r: REnv.(env_t) R).
Context (sigma: forall f, Sig_denote (Sigma f)).
Context (rules: rule_name_t -> rule).
Fixpoint interp_scheduler'
(sched_log: Log)
(s: scheduler)
{struct s} :=
let interp_try rl s1 s2 :=
match interp_rule r sigma sched_log (rules rl) with
| Some l => interp_scheduler' (log_app l sched_log) s1
| None => interp_scheduler' sched_log s2
end in
match s with
| Done => sched_log
| Cons r s => interp_try r s s
| Try r s1 s2 => interp_try r s1 s2
| SPos _ s => interp_scheduler' sched_log s
end.
Definition interp_scheduler (s: scheduler) :=
interp_scheduler' log_empty s.
End Scheduler.
Definition interp_cycle (sigma: forall f, Sig_denote (Sigma f)) (rules: rule_name_t -> rule)
(s: scheduler) (r: REnv.(env_t) R) :=
commit_update r (interp_scheduler r sigma rules s).
End Interp.
Notation interp_args r sigma Gamma sched_log action_log args :=
(interp_args' (@interp_action _ _ _ _ _ _ _ _ r sigma) Gamma sched_log action_log args).