-
Notifications
You must be signed in to change notification settings - Fork 12
/
pipeline_tutorial.v
503 lines (386 loc) · 18.6 KB
/
pipeline_tutorial.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
(*! Tutorial: Simple arithmetic pipeline !*)
Require Import Koika.Frontend.
(*|
===========================================================
A simple arithmetic pipeline and its proof of correctness
===========================================================
In this file we model a simple arithmetic pipeline with two functions `f` and `g`::
input ---(f)--> queue ---(g)--> output
For simplicity, ``input`` and ``output`` will be registers, and we'll update ``input`` at the end of each cycle using an external function ``next_input`` (think of it as an oracle).
Our system will have two rules: ``doF`` and ``doG``, each corresponding to one step of the pipeline.
All that follows is parameterized on a register size ``sz``, set to 32 bits by default.
|*)
Definition sz := pow2 5.
(*|
Setup
=====
We start by defining the system using inductive types to declare its registers and rules and their types:
|*)
Inductive reg_t :=
| input_buffer (** The data source (from which we read inputs) *)
| queue_empty (** A flag indicating whether the queue is empty *)
| queue_data (** The data stored in the queue *)
| output_buffer. (** The data sink (into which we write outputs) *)
Definition R r :=
match r with
| input_buffer => bits_t sz
| queue_empty => bits_t 1
| queue_data => bits_t sz
| output_buffer => bits_t sz
end.
Inductive ext_fn_t :=
| NextInput (** ``NextInput`` steps through the input stream. **)
| F
| G.
Definition Sigma (fn: ext_fn_t) : ExternalSignature :=
match fn with
| NextInput => {$ bits_t sz ~> bits_t sz $}
| F => {$ bits_t sz ~> bits_t sz $}
| G => {$ bits_t sz ~> bits_t sz $}
end.
(*|
Then we declare the rules of the system. Notice how ``doG`` writes at port 0 and ``doF`` reads at port 1. This is because we're building a pipelined queue, not a bypassing one, so we expect clients to pull first and push second. Notice also that each rule is "guarded" by a call to ``fail``; that is, if ``doF`` can't push into the queue because ``doG`` is lagging then ``doF`` won't run at all, and if there's no data ready for ``doG`` to consume because ``doF`` isn't keeping up then ``doG`` won't run.
|*)
Definition _doF : uaction _ _ :=
{{
let v := read0(input_buffer) in
write0(input_buffer, extcall NextInput(v));
let queue_empty := read1(queue_empty) in
if queue_empty then
write1(queue_empty, Ob~0);
write0(queue_data, extcall F(v))
else
fail
}}.
Definition _doG : uaction _ _ :=
{{
let queue_empty := read0(queue_empty) in
if !queue_empty then
let data := read0(queue_data) in
write0(output_buffer, extcall G(data));
write0(queue_empty, Ob~1)
else
fail
}}.
(*|
Rules are written in an untyped language, so we need to typecheck them, which is done using ``tc_rules``:
|*)
Inductive rule_name_t :=
| doF
| doG.
Definition rules : rule_name_t -> rule R Sigma :=
Eval vm_compute in
tc_rules R Sigma
(fun rl => match rl with
| doF => _doF
| doG => _doG
end).
(*|
The last step if to define the system's scheduler. Note that we perform ``doG`` before ``doF`` (intuitively, pipelines run back-to-front: in a sequential model, G removes the data found in the queue and writes it to the output, then F reads the input and writes it in the queue).
|*)
Definition pipeline : scheduler :=
doG |> doF |> done.
(*|
That's it, we've defined our program! There are a few things we can do with it now.
Running our program
===================
Kôika's semantics are deterministic, so we can simply run the program using the interpreter. For this we need to define two provide two additional pieces: the value of the registers at the beginning of the cycle (``r``), and a model of the external functions (``σ``).
For the registers, we start with an empty queue, the input set to 32'd1, and the output set to 32'd0. The notation ``#{ reg => val #}`` indicates that register ``reg`` maps to value ``val``. Notations like ``Ob~1~0~0~1`` denote bitstring constants: for example, ``Ob~0~1~1`` is the Kôika equivalent of ``3'b011`` in Verilog.
|*)
(** We need [reg_t] to be finite to construct register maps. *)
Instance FiniteType_reg_t : FiniteType reg_t := _.
Definition r : ContextEnv.(env_t) R :=
#{ input_buffer => Bits.of_nat 32 1;
queue_empty => Ob~1;
queue_data => Bits.of_nat 32 0;
output_buffer => Bits.of_nat 32 0 }#.
(*|
For the external functions, we take something simple: ``f`` multiplies its input by 5, ``g`` adds one to its input, and ``next_input`` models the sequence of odd numbers ``1, 3, 5, 7, 9, …``:
|*)
Definition σ fn : Sig_denote (Sigma fn) :=
match fn with
| NextInput => fun x => x +b (Bits.of_nat 32 2)
| F => fun x => Bits.slice 0 32 (x *b (Bits.of_nat 32 5))
| G => fun x => x +b (Bits.of_nat 32 1)
end.
(*|
In the first step of execution ``doG`` does not execute and ``doF`` reads 1 from the input and writes 5 (32'b101) in the queue; the input then becomes 3 (32'b11).
The function ``interp_cycle`` computes this update: it returns a map of register values.
|*)
Compute (interp_cycle σ rules pipeline r). (* .unfold *)
(*|
Here is an infinite stream capturing the initial state of the system and all snapshots of the system after each execution:
|*)
Definition system_states :=
Streams.coiterate (interp_cycle σ rules pipeline) r.
(*|
Forcing this infinite stream simulates the whole system directly inside Coq: for example, we can easily extract the first few inputs that the system processes, and the first few outputs that it produces:
- Inputs (``1``; ``3``; ``5``; ``7``; ``9``; ``…``):
..
|*)
Compute (Streams.firstn 5
(Streams.map (fun r => r.[input_buffer])
system_states)). (* .unfold *)
Compute (Streams.firstn 5
(Streams.map (fun r => @Bits.to_nat 32 r.[input_buffer])
system_states)). (* .unfold *)
(*|
- Outputs (``1 * 5 + 1 = 6``; ``3 * 5 + 1 = 16``; ``5 * 5 + 1 = 26``; ``…``); the first two zeros correspond pipeline's initial lag (the first 0 is the initial state; the second, the result of running one cycle; after that, the output is updated at every cycle):
..
|*)
Compute (Streams.firstn 5
(Streams.map (fun r => r.[output_buffer])
system_states)). (* .unfold *)
Compute (Streams.firstn 5
(Streams.map (fun r => @Bits.to_nat 32 r.[output_buffer])
system_states)). (* .unfold *)
(*|
Simulating this way is convenient for exploring small bits of code; for large designs, we have a compiler that generates optimized (but readable!) C++ that simulates Kôika designs 4 to 5 orders of magnitude faster than running directly in Coq does (a small example like this one runs hundreds of millions of simulated cycles per second in C++, and thousands in Coq):
|*)
Time Compute (@Bits.to_nat 32
(Streams.Str_nth 1000 system_states).[output_buffer]). (* .unfold *)
(*|
Running individual rules
------------------------
We can also simulate at finer granularity: for example, here is what happens for each rule during the first cycle (each step produces a log of events that happen as part of this rule's execution):
- ``doG`` fails, because the queue is empty:
..
|*)
Compute (interp_rule r σ log_empty (rules doG)). (* .unfold *)
(*|
- ``doF`` succeeds and updates the queue and the input buffer:
..
|*)
Compute (interp_rule r σ log_empty (rules doF)). (* .unfold *)
(*|
In later cycles (here, cycle 3), ``doG`` succeeds as well and writes 1 to (port 0 of) ``queue_empty``…
|*)
Compute (let r := Streams.Str_nth 3 system_states in
interp_rule r σ log_empty (rules doG)). (* .unfold *)
(*|
… and ``doF`` sets (port 1 of) ``queue_empty`` back to 0 as it refills the queue:
|*)
Compute (let r := Streams.Str_nth 3 system_states in
let logG := interp_rule r σ log_empty (rules doG) in
interp_rule r σ (must logG) (rules doF)). (* .unfold *)
(*|
Generating circuits
===================
Generating a circuit is just a matter of calling the compiler, which is a Gallina function too:
|*)
(** ``external`` is an escape hatch used to inject arbitrary Verilog
into a design; we don't use it here **)
Definition external (r: rule_name_t) := false.
Definition circuits :=
compile_scheduler rules external pipeline.
(*|
For printing circuits, we don't recommend using Coq's ``Compute``: our representation of circuits uses physical equality to track shared subcircuits, but Coq's printer doesn't respect sharing when printing, and hence generated circuits look giant. Instead, we recommend looking at the generated Verilog or Dot graphs (use ``make examples/_objects/pipeline_tutorial.v/`` in the repository's root and navigate to ``examples/_objects/pipeline_tutorial.v/pipeline_tutorial.v`` to see the generated Verilog code, which uses internally instantiated modules for ``F`` and ``G`` and signature wires for ``NextInput``).
We can, however, easily compute the results produced by a circuit, either after one cycle:
|*)
Compute (interp_circuits σ circuits (lower_r r)). (* .unfold *)
(*|
… or after multiple cycles:
|*)
Definition circuit_states :=
Streams.coiterate (interp_circuits σ circuits) (lower_r r).
Compute (Streams.firstn 5
(Streams.map (fun r => r.[output_buffer])
circuit_states)). (* .unfold *)
Compute (Streams.firstn 5
(Streams.map (fun r => @Bits.to_nat 32 r.[output_buffer])
circuit_states)). (* .unfold *)
(*|
Proving properties
==================
The compiler is proven correct, so we know that running a circuit will always produce the same results as running the original Kôika design. But we haven't proved anything yet about our Kôika design, so it could be full of bugs. Here we're interested in ruling out two types of bugs: function correctness bugs and timing bugs.
The first class of bugs would cause our design to compute something different from `g ∘ f`. The second class of bug would cause it to stutter or lag, taking more than one cycle to produce each output.
|*)
Require Koika.CompactSemantics.
Section Correctness.
(*|
We start by quantifying over ``σ``, the model of external functions. This matters because we want to prove our design correct regardless of the concrete functions that we plug in:
|*)
Context (σ: forall f, Sig_denote (Sigma f)).
(*|
With this definition, ``sigma F: bits_t 32 → bits_t 32`` is the model of `f`, ``sigma G`` is the model of `g`, and ``sigma NextInput`` is the model of the `next_input` oracle.
.. coq:: none
|*)
Arguments id _ _ / : assert.
Arguments env_t: simpl never.
Arguments vect: simpl never.
Arguments may_read /.
Arguments may_write /.
Opaque CompactSemantics.interp_cycle.
(* FIXME remove these notations *)
Notation "0b0" := {| vhd := false; vtl := _vect_nil |}.
Notation "0b1" := {| vhd := true; vtl := _vect_nil |}.
Import StreamNotations.
(*|
Let's start by stating a specification that will rule both kinds of bugs. In this example we completely characterize the pipeline (we have examples that use weaker characterizations such as invariants, but we haven't written a tutorial for them yet). Note how we quantify over ``r``: as we'll see later, our design will work for any starting state, as long as the queue starts empty.
|*)
Section Spec.
Context (r: ContextEnv.(env_t) R).
(*|
Here is the stream of inputs consumed by the spec: we just iterate ``NextInput`` on the original value ``r.[input_buffer]``.
|*)
Definition spec_inputs :=
Streams.coiterate (σ NextInput) r.[input_buffer].
(*|
Here is the expected stream of outputs, which we call “observations”. We only expect outputs to start becoming available after completing two cycles, so we simply state that the value in ``output_buffer`` should be unchanged until then:
|*)
Definition spec_observations :=
let composed x := σ G (σ F x) in
r.[output_buffer] ::: (* Initial value *)
r.[output_buffer] ::: (* Unchanged after one cycle *)
Streams.map composed spec_inputs. (* Actual outputs *)
(*|
Here is the actual stream of states produced by the implementation, which we previously called “system_states”. Note that we use the ``CompactSemantics`` module, which uses a more explicit representation of logs that plays more smoothly with abstract interpretation (instead of storing lists of events, it stores a record indicating whether we've read or written at port 0 and 1).
.. FIXME: use regular semantics and introduce compact ones in the proof.
|*)
Definition impl_trace : Streams.Stream (ContextEnv.(env_t) R) :=
Streams.coiterate
(CompactSemantics.interp_cycle σ rules pipeline)
r.
(*|
Finally, here is the actual stream of observations produced by the implementation:
|*)
Definition impl_observations :=
Streams.map (fun r => r.[output_buffer]) impl_trace.
End Spec.
(*|
The approach we'll use in this proof is to give a complete characterization of two cycles of the pipeline, and then use k-induction for `k = 2` to generalize it into a complete characterization of the system for any number of cycles. Here's how that looks:
|*)
Section Proofs.
(*|
This definition captures what it means to execute one cycle:
|*)
Definition cycle (r: ContextEnv.(env_t) R) :=
CompactSemantics.interp_cycle σ rules pipeline r.
(*|
Here is our two-cycle characterization: if we execute our circuit twice, ``input_buffer`` takes two steps of ``NextInput``, ``queue_empty`` becomes false (regardless of where we start, the queue will be filled). ``queue_data`` contains ``f`` of the input updated by the first cycle, and ``output`` buffer contains the correct output.
|*)
Definition phi2 (r: ContextEnv.(env_t) R)
: ContextEnv.(env_t) R :=
#{ input_buffer => σ NextInput (σ NextInput r.[input_buffer]);
queue_empty => Ob~0;
queue_data => σ F (σ NextInput r.[input_buffer]);
output_buffer => σ G (σ F r.[input_buffer]) }#.
(*|
Proving this characterization is just a matter of abstract interpretation:
|*)
Lemma phi2_correct :
forall r, cycle (cycle r) = phi2 r.
Proof.
intros r; unfold cycle.
(*|
We start by unfolding the inner call to ``interp_cycle``, which yields a characterization that branches on ``r.[queue_empty]``:
|*)
abstract_simpl r. (* .unfold *)
(*|
There are three cases: ``negb (Bits.single r.[queue_empty])``, in which both ``doG`` and ``doF`` execute; ``Bits.single r.[queue_empty]``, in which only ``doF`` executes; and an impossible third case in which neither rule would have executed.
To explore these cases we do a case split on ``r.[queue_empty]``, and simplify to show that they lead to the same result:
|*)
destruct (Bits.single r.[queue_empty]); abstract_simpl. (* .unfold *)
all: reflexivity.
Qed.
(*|
With this done, we can now prove a stronger characterization that holds for any number of cycles:
|*)
Definition phi_iterated n
(r: ContextEnv.(env_t) R)
: ContextEnv.(env_t) R :=
let input := r.[input_buffer] in
#{ input_buffer => iterate (S (S n)) (σ NextInput) input;
queue_empty => Ob~0;
queue_data => σ F (iterate (S n) (σ NextInput) input);
output_buffer => σ G (σ F (iterate n (σ NextInput) input)) }#.
(*|
The proof is a two-induction (captured by the ``Div2.ind_0_1_SS`` lemma); it tells us what the registers contain after ``n + 2`` cycles of the pipeline:
|*)
Lemma phi_iterated_correct :
forall r n,
iterate (S (S n)) cycle r =
phi_iterated n r.
Proof.
intros r n.
rewrite !iterate_S_acc, phi2_correct.
revert n; apply Div2.ind_0_1_SS; simpl. (* .unfold *)
- reflexivity.
- unfold cycle; reflexivity.
- intros n IH; simpl in IH; rewrite IH; reflexivity.
Qed.
(*|
And this is enough to complete our proof! We'll manually match up the first two states, then use our characterization. We need to assume that the queue is empty to begin with, otherwise we'd start producing unexpected output in the first cycle; this hypothesis comes into play when matching up the second element of the traces.
|*)
Theorem correct_pipeline:
forall (r: ContextEnv.(env_t) R),
r.[queue_empty] = Ob~1 ->
Streams.EqSt (impl_observations r) (spec_observations r).
Proof.
intros r Hqueue_empty.
unfold impl_observations, spec_observations, impl_trace.
apply Streams.ntheq_eqst; intros [ | [ | n ] ]; simpl;
unfold spec_inputs;
rewrite ?Streams.Str_nth_0, ?Streams.Str_nth_S,
?Streams.Str_nth_map, ?Streams.Str_nth_coiterate.
- (* Initial state *) (* .unfold *)
simpl.
reflexivity.
- (* After one cycle *) (* .unfold *)
simpl.
abstract_simpl.
rewrite Hqueue_empty; reflexivity.
- (* After two cycles *) (* .unfold *)
rewrite phi_iterated_correct.
simpl.
reflexivity.
Qed.
End Proofs.
End Correctness.
(*|
Extracting to Verilog and C++
=============================
For this final piece, we need to specify which C++ and Verilog functions ``F``, ``G``, and ``NextInput`` correspond to; we do so by constructing a ``package``, which contains all relevant information:
|*)
Definition cpp_extfuns := "class extfuns {
public:
static bits<32> next_input(bits<32> st) {
return st + bits<32>{2};
}
static bits<32> f(bits<32> x) {
return prims::slice<0, 32>(x * bits<32>{5});
}
static bits<32> g(bits<32> x) {
return x + bits<32>{1};
}
};".
Definition ext_fn_names fn :=
match fn with
| NextInput => "next_input"
| F => "f"
| G => "g"
end.
Definition package :=
{| ip_koika := {| koika_reg_types := R;
koika_reg_init reg := r.[reg];
koika_ext_fn_types := Sigma;
koika_rules := rules;
koika_rule_external := external;
koika_scheduler := pipeline;
koika_module_name := "pipeline_tutorial" |};
ip_sim := {| sp_ext_fn_specs fn :=
{| efs_name := ext_fn_names fn;
efs_method := false |};
sp_prelude := Some cpp_extfuns |};
ip_verilog := {| vp_ext_fn_specs fn :=
{| efr_name := ext_fn_names fn;
efr_internal :=
match fn with
| F | G => true
| NextInput => false
end |} |} |}.
(*|
This last bit registers our program, which allows the build system to locate it:
|*)
Definition prog := Interop.Backends.register package.
Extraction "pipeline_tutorial.ml" prog.