-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathExtraction.v
155 lines (139 loc) · 8.04 KB
/
Extraction.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
Require Export List String Ascii BinInt BinNat.
Require Export Kami.Syntax Kami.Compiler.CompilerSimple Kami.Compiler.Compiler Kami.Compiler.Rtl Kami.LibStruct Kami.Compiler.UnverifiedIncompleteCompiler.
Require Import Kami.Notations.
Require Coq.extraction.Extraction.
Require Export ExtrHaskellBasic ExtrHaskellNatInt ExtrHaskellString ExtrHaskellZInteger.
Extraction Language Haskell.
Set Extraction Optimize.
Set Extraction KeepSingleton.
Unset Extraction AutoInline.
Extract Inductive sigT => "(,)" ["(,)"].
Extract Inductive Fin.t => "CustomExtract.EFin" ["CustomExtract.fin0" "CustomExtract.finS"] "CustomExtract.finRec".
Extract Inductive N => "Prelude.Integer" ["0" "(\x -> x)"] "(\fn0 fnpos x -> if x Prelude.== 0 then fn0 () else fnpos x)".
Extract Inlined Constant fst => "Prelude.fst".
Extract Inlined Constant snd => "Prelude.snd".
Extract Inlined Constant projT1 => "Prelude.fst".
Extract Inlined Constant projT2 => "Prelude.snd".
Extract Inlined Constant map => "Prelude.map".
Extract Inlined Constant List.concat => "Prelude.concat".
Extract Inlined Constant String.concat => "Data.List.intercalate".
Extract Inlined Constant mod2 => "Prelude.odd".
Extract Constant nat_cast => "(\_ _ x -> x)".
Extract Inlined Constant length => "Prelude.length".
Extract Inlined Constant Datatypes.length => "Prelude.length".
Extract Constant Nat.div2 => "(`Prelude.div` 2)".
Extract Constant Nat.log2 => "(\x -> Prelude.floor (Prelude.logBase 2 (Prelude.fromIntegral x)))".
Extract Constant Nat.log2_up => "(\x -> Prelude.ceiling (Prelude.logBase 2 (Prelude.fromIntegral x)))".
Extract Constant List.fold_left => "(\f bs a -> Data.List.foldl' f a bs)".
Extract Constant natToWord => "(\sz n -> Prelude.toInteger n)".
Extract Constant wordToNat => "(\_ -> Prelude.fromIntegral)".
Extract Constant sumSizes => "(\n f -> Prelude.sum (Prelude.map (\i -> f (n Prelude.-1,i)) [0..(n Prelude.-1)]))".
Extract Constant nth_Fin => "(\xs (_,i) -> xs Prelude.!! i)".
Extract Constant nth_Fin_map2 => "(\_ _ _ x -> x)".
Extract Constant getFins => "(\n -> Prelude.map ((,) (n Prelude.- 1)) [0..(n Prelude.- 1)])".
Extract Constant Fin.to_nat => "(\_ (_,i) -> i)".
Extract Constant Fin.cast => "(\_ x _ -> x)".
Extract Constant Fin.of_nat_lt => "(\i n -> (n Prelude.- 1,i))".
Extract Constant Fin_eq_dec => "(\_ x y -> x Prelude.== y)".
Extract Inlined Constant getBool => "Prelude.id".
Extract Inlined Constant String.append => "(Prelude.++)".
Extract Constant ZToWord => "(\n x -> Prelude.mod x (2 Prelude.^ n))".
Extract Inlined Constant NToWord => "(\_ x -> x)".
Extract Constant wones => "(\n -> 2 Prelude.^ n Prelude.- 1)".
Extract Constant wadd => "(\_ x y -> x Prelude.+ y)".
Extract Constant wsub => "(\_ x y -> x Prelude.- y)".
Extract Constant wor => "(\_ x y -> x Data.Bits..|. y)".
Extract Constant wand => "(\_ x y -> x Data.Bits..&. y)".
Extract Constant wxor => "(\_ -> Data.Bits.xor)".
Extract Constant wnot => "(\_ -> Data.Bits.complement)".
Extract Constant wuxor => "(\_ x -> Prelude.odd (Data.Bits.popCount x))".
Extract Constant wmul => "(\_ x y -> x Prelude.* y)".
Extract Constant wdiv => "(\_ -> Prelude.div)".
Extract Constant wmod => "(\_ -> Prelude.mod)".
Extract Constant wslu => "(\_ x n -> Data.Bits.shiftL x (Prelude.fromIntegral n))".
Extract Constant wsru => "(\_ x n -> Data.Bits.shiftR x (Prelude.fromIntegral n))".
Extract Constant weqb => "(\_ -> (Prelude.==))".
Extract Constant wuand => "(\n x -> x Prelude.== 2 Prelude.^ n)".
Extract Constant wuor => "(\_ x -> Prelude.not (x Prelude.== 0))".
Extract Constant wltu => "(\_ -> (Prelude.<))".
Extract Constant truncMsb => "(\msb _ x -> Data.Bits.shiftR x msb)".
Extract Inlined Constant Z.pow => "(Prelude.^)".
Extract Inlined Constant Z.of_nat => "Prelude.toInteger".
Extract Inlined Constant Z.ltb => "(Prelude.<)".
Extract Inlined Constant Z.opp => "Prelude.negate".
Extract Inlined Constant Z.div => "Prelude.div".
Extract Inlined Constant Z.modulo => "Prelude.mod".
Extract Inlined Constant N.succ_pos => "(\x -> x Prelude.+ 1)".
Extract Inlined Constant N.add => "(Prelude.+)".
Extract Inlined Constant N.sub => "(Prelude.-)".
Extract Inlined Constant N.mul => "(Prelude.*)".
Extract Inlined Constant N.eqb => "(Prelude.==)".
Extract Inlined Constant N.ltb => "(Prelude.<)".
Extract Inlined Constant N.of_nat => "Prelude.toInteger".
Section Ty.
Variable ty: Kind -> Type.
Local Open Scope kami_expr.
Definition predPack k (pred: Bool @# ty) (val: k @# ty) :=
(IF pred
then pack val
else $0).
Definition orKind k (ls: list (Bit (size k) @# ty)) := unpack k (Kor ls).
Definition predPackOr k (ls: list ((Bool @# ty) * (k @# ty))) := ((@Kor _ Bool) (map fst ls), orKind k (map (fun '(p, v) => predPack p v) ls)).
Definition createWriteRq ty (idxNum num: nat) (k: Kind) (idx: Bit (Nat.log2_up idxNum) @# ty) (val: Array num k @# ty): WriteRq (Nat.log2_up idxNum) (Array num k) @# ty :=
STRUCT { "addr" ::= idx ;
"data" ::= val }.
Definition createWriteRqMask ty (idxNum num: nat) (k: Kind) (idx: Bit (Nat.log2_up idxNum) @# ty) (val: Array num k @# ty) (mask: Array num Bool @# ty): WriteRqMask (Nat.log2_up idxNum) num k @# ty :=
STRUCT { "addr" ::= idx ;
"data" ::= val ;
"mask" ::= mask }.
Definition pointwiseIntersectionNoMask (idxNum num: nat) (k: Kind)
(readPred: Bool @# ty)
(readAddr: Bit (Nat.log2_up idxNum) @# ty)
(writePred: Bool @# ty) (writeRq: WriteRq (Nat.log2_up idxNum) (Array num k) @# ty)
: Array num (Maybe k) @# ty
:= BuildArray
(fun i =>
let readAddr_i := readAddr + $(proj1_sig (Fin.to_nat i)) in
STRUCT { "valid" ::= (readPred
&& writePred
&& (writeRq @% "addr" <= readAddr_i)
&& (readAddr_i < writeRq @% "addr" + $num));
"data" ::= (writeRq @% "data")@[readAddr - writeRq @% "addr" + $(proj1_sig (Fin.to_nat i))] } : Maybe k @# ty).
Definition pointwiseIntersectionMask (idxNum num: nat) (k: Kind)
(readPred: Bool @# ty)
(readAddr: Bit (Nat.log2_up idxNum) @# ty)
(writePred: Bool @# ty) (writeRq: WriteRqMask (Nat.log2_up idxNum) num k @# ty)
: Array num (Maybe k) @# ty
:= BuildArray
(fun i =>
let readAddr_i := readAddr + $(proj1_sig (Fin.to_nat i)) in
STRUCT { "valid" ::= (readPred
&& writePred
&& ((writeRq @% "mask")@[readAddr - writeRq @% "addr" + $(proj1_sig (Fin.to_nat i))])
&& (writeRq @% "addr" <= readAddr_i)
&& (readAddr_i < writeRq @% "addr" + $num));
"data" ::= (writeRq @% "data")@[readAddr - writeRq @% "addr" + $(proj1_sig (Fin.to_nat i))] } : Maybe k @# ty).
Definition pointwiseIntersection (idxNum num: nat) (k: Kind) (isMask: bool)
(readPred: Bool @# ty)
(readAddr: Bit (Nat.log2_up idxNum) @# ty)
(writePred: Bool @# ty) (writeRq: if isMask
then WriteRqMask (Nat.log2_up idxNum) num k @# ty
else WriteRq (Nat.log2_up idxNum) (Array num k) @# ty)
: Array num (Maybe k) @# ty :=
match isMask return (if isMask
then WriteRqMask (Nat.log2_up idxNum) num k @# ty
else WriteRq (Nat.log2_up idxNum) (Array num k) @# ty) -> Array num (Maybe k) @# ty
with
| true => fun writeRq =>
pointwiseIntersectionMask idxNum readPred readAddr writePred writeRq
| false => fun writeRq =>
pointwiseIntersectionNoMask idxNum readPred readAddr writePred writeRq
end writeRq.
Definition pointwiseBypass (num: nat) (k: Kind) (bypass: Array num (Maybe k) @# ty) (resp: Array num k @# ty)
: Array num k @# ty :=
BuildArray
(fun i => (IF (ReadArrayConst bypass i) @% "valid"
then (ReadArrayConst bypass i) @% "data"
else ReadArrayConst resp i)).
Local Close Scope kami_expr.
End Ty.