-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtydi-example.test.scala
257 lines (206 loc) · 8.39 KB
/
tydi-example.test.scala
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
//> using scala "2.13.14"
//> using dep "com.github.rameloni::tywaves-chisel-api:0.4.2-SNAPSHOT"
//> using dep "nl.tudelft::tydi-chisel::0.1.0"
//> using plugin "org.chipsalliance:::chisel-plugin:6.4.0"
//> using options "-unchecked", "-deprecation", "-language:reflectiveCalls", "-feature", "-Xcheckinit", "-Xfatal-warnings", "-Ywarn-dead-code", "-Ywarn-unused", "-Ymacro-annotations"
//> using dep "org.scalatest::scalatest:3.2.18"
// DO NOT EDIT THE ORTHER OF THESE IMPORTS (it will be solved in future versions)
import tywaves.simulator._
import tywaves.simulator.ParametricSimulator._
import tywaves.simulator.simulatorSettings._
import chisel3._
// _root_ disambiguates from package chisel3.util.circt if user imports chisel3.util._
//import _root_.circt.stage.ChiselStage
import org.scalatest.funspec.AnyFunSpec
import org.scalatest.matchers.should.Matchers
import nl.tudelft.tydi_chisel._
import chisel3.util.Counter
object MyTypes {
/** Bit(64) type, defined in pipelineSimple_types */
def generated_0_7_Tc5SbYQz_27 = UInt(64.W)
assert(this.generated_0_7_Tc5SbYQz_27.getWidth == 64)
/** Bit(64) type, defined in pipelineSimple_types */
def generated_0_7_d3p7qsW3_29 = SInt(64.W)
assert(this.generated_0_7_d3p7qsW3_29.getWidth == 64)
}
/**
* Group element, defined in pipelineSimple_types. A composite type (like a
* struct) that contains a value associated with a timestamp.
*/
class NumberGroup extends Group {
val time = MyTypes.generated_0_7_Tc5SbYQz_27
val value = MyTypes.generated_0_7_d3p7qsW3_29
}
/**
* Group element, defined in pipelineSimple_types. A composite type (like a
* struct) that represents the stats of the implemented algorithm.
*/
class Stats extends Group {
val average = MyTypes.generated_0_7_Tc5SbYQz_27
val max = MyTypes.generated_0_7_Tc5SbYQz_27
val min = MyTypes.generated_0_7_Tc5SbYQz_27
val sum = MyTypes.generated_0_7_Tc5SbYQz_27
}
/** Stream, defined in pipelineSimple_types. */
class Generated_0_30_soHLkh1p_30
extends PhysicalStreamDetailed(e = new Stats, n = 1, d = 1, c = 1, r = false, u = Null())
object Generated_0_30_soHLkh1p_30 {
def apply(): Generated_0_30_soHLkh1p_30 = Wire(new Generated_0_30_soHLkh1p_30())
}
/** Stream, defined in pipelineSimple_types. */
class Generated_0_36_xeHH4woS_24
extends PhysicalStreamDetailed(e = new NumberGroup, n = 1, d = 2, c = 1, r = false, u = Null())
object Generated_0_36_xeHH4woS_24 {
def apply(): Generated_0_36_xeHH4woS_24 = Wire(new Generated_0_36_xeHH4woS_24())
}
/** Bit(64), defined in pipelineSimple_types. */
class Generated_0_7_Tc5SbYQz_27 extends BitsEl(64.W)
/** Bit(64), defined in pipelineSimple_types. */
class Generated_0_7_d3p7qsW3_29 extends BitsEl(64.W)
/**
* Streamlet, defined in pipelineSimple. Top level interface.
*/
class PipelineSimple_interface extends TydiModule {
/** Stream of [[in]] with input direction. */
val inStream = Generated_0_36_xeHH4woS_24().flip
/** IO of [[inStream]] with input direction. */
val in = inStream.toPhysical
/** Stream of [[out]] with output direction. */
val outStream = Generated_0_30_soHLkh1p_30()
/** IO of [[outStream]] with output direction. */
val out = outStream.toPhysical
}
/**
* Streamlet, defined in pipelineSimple. Interface for the agg function.
*/
class Reducer_interface extends TydiModule {
/** Stream of [[in]] with input direction. */
val inStream = Generated_0_36_xeHH4woS_24().flip
/** IO of [[inStream]] with input direction. */
val in = inStream.toPhysical
/** Stream of [[out]] with output direction. */
val outStream = Generated_0_30_soHLkh1p_30()
/** IO of [[outStream]] with output direction. */
val out = outStream.toPhysical
}
/**
* Streamlet, defined in pipelineSimple. Interface for the non negative filter:
* df.filter(col("value") >= 0).
*/
class Instance_NonNegativeFilter_interface_RefToVargenerated_0_36_KLOZ7sts_25_1 extends TydiModule {
/** Stream of [[in]] with input direction. */
val inStream = Generated_0_36_xeHH4woS_24().flip
/** IO of [[inStream]] with input direction. */
val in = inStream.toPhysical
/** Stream of [[out]] with output direction. */
val outStream = Generated_0_36_xeHH4woS_24()
/** IO of [[outStream]] with output direction. */
val out = outStream.toPhysical
}
/**
* Implementation, defined in pipelineSimple. Implementation of
* df.filter(col("value") >= 0).
*/
class NonNegativeFilter extends Instance_NonNegativeFilter_interface_RefToVargenerated_0_36_KLOZ7sts_25_1 {
// Filtered only if the value is non-negative
// inStream.valid tells us if the input is valid
private val canPass: Bool = inStream.el.value >= 0.S && inStream.valid
// Connect inStream to outStream
// This is equivalent of connecting al the fields of the two streams
// Every future assignment will overwrite the previous one
outStream := inStream
// Always ready to accept input
inStream.ready := true.B
// if (canPass) then { it can go out } else { it is not forwarded }
outStream.valid := canPass && outStream.ready
outStream.strb := inStream.strb(0) && canPass
// All the other signals are forwarded
// outStream.stai := inStream.stai
// outStream.endi := inStream.endi
// outStream.last := inStream.last
// outStream.el := inStream.el
}
/**
* Implementation, defined in pipelineSimple. Top level implementation. It
* instantiates the subcomponents and connects them together.
*/
class PipelineSimple extends PipelineSimple_interface {
inStream := DontCare
outStream := DontCare // Probably not used
// Modules
private val filter = Module(new NonNegativeFilter)
private val reducer = Module(new Reducer)
// Connections
filter.in := in // This will overload the inStream := in
reducer.in := filter.out
out := reducer.out // This will overload the out := outStream -> that's why outStream is not used
}
/**
* Implementation, defined in pipelineSimple. Implementation of the agg
* function.
*/
class Reducer extends Reducer_interface {
// Set the data width to 64 bits, such as the [[MyTypes]] types
private val dataWidth = 64.W
// Computing min and max
private val maxVal: BigInt = BigInt(Long.MaxValue) // Must work with BigInt or we get an overflow
private val cMin: UInt = RegInit(maxVal.U(dataWidth)) // REG for the min value
private val cMax: UInt = RegInit(0.U(dataWidth)) // REG for the max value
// Computing the sum
private val cSum: UInt = RegInit(0.U(dataWidth)) // REG for the sum
// Computing the avg
private val nValidSamples: Counter = Counter(Int.MaxValue) // The number of samples received (valid && strb(0))
private val nSamples: Counter = Counter(Int.MaxValue) // The number of samples received (valid)
// Set the streams IN ready and OUT valid signals
// inReady = if (maxVal > 0) { true.B } else { false.B}
inStream.ready := (if (maxVal > 0) true.B else false.B)
// inStream.ready := true.B
// The output is valid only if we have received at least one sample
outStream.valid := nSamples.value > 0.U
// When a value is received
when(inStream.valid) {
// Get the value from th input stream
val value = inStream.el.value.asUInt
nSamples.inc()
// Check the strb line and perform the updates
when(inStream.strb(0)) {
cMin := cMin min value
cMax := cMax max value
cSum := cSum + value
nValidSamples.inc()
}
}
// Set the output stream
outStream.el.sum := cSum
outStream.el.min := cMin
outStream.el.max := cMax
outStream.el.average := Mux(nValidSamples.value > 0.U, cSum / nValidSamples.value, 0.U)
// Set the output stream control signals:
// they are fixed since the sum, min, max and average are updated every cycle
// and they hold the same value if no change is performed
outStream.strb := 1.U
outStream.stai := 0.U
outStream.endi := 1.U
outStream.last(0) := inStream.last(0)
}
class PipelineSimpleTest extends AnyFunSpec with Matchers {
describe("ParametricSimulator") {
it("runs Pipeline Simple correctly") {
simulate(new PipelineSimple, Seq(VcdTrace, SaveWorkdirFile("pipelineSimpleWorkdir"))) {
dut => dut.clock.step()
}
}
}
describe("TywavesSimulator") {
it("runs Pipeline Simple correctly") {
import TywavesSimulator._
simulate(
new PipelineSimple(),
Seq(VcdTrace, WithTywavesWaveforms(true)),
simName = "runs_GCD_correctly_launch_tywaves",
) {
dut => dut.clock.step()
}
}
}
}