-
Notifications
You must be signed in to change notification settings - Fork 31
/
Copy pathGLBCluster.scala
410 lines (387 loc) · 18.7 KB
/
GLBCluster.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
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
package dla.cluster
import chisel3._
import chisel3.util._
import dla.pe.{CSCStreamIO, MCRENFConfig, StreamBitsIO}
class GLBCluster(debug: Boolean) extends Module with ClusterSRAMConfig with GNMFCS2Config {
val io: GLBClusterIO = IO(new GLBClusterIO)
protected val iSRAMs: Seq[InActSRAMBank] = Seq.fill(inActSRAMNum){Module(new InActSRAMBank(debug))}
iSRAMs.zipWithIndex.foreach({ case (o, i) => o.suggestName(s"inActSRAM$i")})
protected val iSRAMsIO: Seq[InActSRAMBankIO] = iSRAMs.map(x => x.io)
protected val pSRAMs: Seq[PSumSRAMBank] = Seq.fill(pSumSRAMNum){Module(new PSumSRAMBank(pSumSRAMSize, psDataWidth, debug))}
pSRAMs.zipWithIndex.foreach({ case (o, i) => o.suggestName(s"pSumSRAM$i")})
protected val pSRAMsIO: Seq[PSumSRAMBankIO] = pSRAMs.map(x => x.io)
protected val theTopCtrls = Seq(io.ctrlPath.inActIO, io.ctrlPath.pSumIO)
protected val theSRAMsCtrl = Seq(iSRAMsIO.map(x => x.ctrlPath), pSRAMsIO.map(x => x.ctrlPath))
protected val theSRAMsNum = Seq(inActSRAMNum, pSumSRAMNum)
// connections of inAct and PSum
for (i <- 0 until 2) { // inAct and PSum
// connections of control path
for (k <- 0 until 2) { // read and write
// inner connections
for (j <- 0 until theSRAMsNum(i)) {
val topCtrlSeq = Seq(theTopCtrls(i)(j).readIO, theTopCtrls(i)(j).writeIO)
val sramCtrlSeq = Seq(theSRAMsCtrl(i)(j).readIO, theSRAMsCtrl(i)(j).writeIO)
sramCtrlSeq(k).enable := topCtrlSeq(k).enable
if (i == 0 && k == 1) { // inActWrite.adr don't need
sramCtrlSeq(k).adr := DontCare
} else {
sramCtrlSeq(k).adr := topCtrlSeq(k).adr
}
if (i == 1 && k == 0) { // pSum.read.done wires, can be seen from the GLB controller
topCtrlSeq(k).done := DontCare
} else {
topCtrlSeq(k).done := sramCtrlSeq(k).done
}
}
}
}
// connections of data path
io.dataPath.weightIO.foreach(x => x.inIOs <> x.outIOs)
io.dataPath.inActIO.zip(iSRAMsIO).foreach({ case (dataIO, sramIO) => dataIO <> sramIO.dataPath})
io.dataPath.pSumIO.zip(pSRAMsIO).foreach({ case (dataIO, sramIO) =>
dataIO <> sramIO.dataPath
})
pSRAMsIO.zipWithIndex.foreach({ case (pSumIO, idx) =>
pSumIO.dataPath <> io.dataPath.pSumIO(idx)
})
// connections of debugIO
if (debug) {
io.debugIO.inActDebugIO.zip(iSRAMsIO).foreach({ case (topDebug, sram) => topDebug <> sram.debugIO})
io.debugIO.pSumDebugIO.zip(pSRAMsIO).foreach({ case (topDebug, sram) => topDebug <> sram.debugIO})
io.debugIO.oneInActSRAMDone <> theSRAMsCtrl.head.map(x => x.readIO.done)
io.debugIO.onePSumSRAMDone <> theSRAMsCtrl(1).map(x => Mux(x.readIO.enable, x.readIO.done, x.writeIO.done)) // so that's the wire not reg done
} else {
io.debugIO <> DontCare
}
}
class PSumSRAMBank(private val theSRAMSize: Int, private val theDataWidth: Int, debug: Boolean)
extends SRAMCommon(theSRAMSize, theDataWidth) with MCRENFConfig {
theSRAM.suggestName("onePSumSRAMBank")
val io: PSumSRAMBankIO = IO(new PSumSRAMBankIO)
// write logic
writeLogic(io.dataPath.inIOs, io.ctrlPath.writeIO.enable, io.ctrlPath.writeIO.adr)
protected val writeCounter: UInt = RegInit(0.U(log2Ceil(pSumOneSPadNum).W))
writeCounter.suggestName("writeCounter")
protected val writeDone: Bool = writeCounter === (pSumOneSPadNum - 1).U && io.dataPath.inIOs.fire()
writeDone.suggestName("writeDone")
writeCounter := Mux(writeDone, 0.U, Mux(io.dataPath.inIOs.fire(), writeCounter + 1.U, writeCounter))
// read logic
// only use ready to control the read progress, so ready signal need to keep at least two cycles
readLogic(io.dataPath.outIOs, io.ctrlPath.readIO.enable, io.ctrlPath.readIO.adr, false.B)
io.ctrlPath.writeIO.done := writeDone
io.ctrlPath.readIO.done := DontCare
// debug io
if (debug) {
debugLogic(io.debugIO, io.ctrlPath.readIO.enable)
} else {
io.debugIO <> DontCare
}
}
abstract class SRAMCommon(private val theSRAMSize: Int, private val theDataWidth: Int) extends Module {
protected val theSRAM: SyncReadMem[UInt] = SyncReadMem(theSRAMSize, UInt(theDataWidth.W), SyncReadMem.ReadFirst)
// SRAM read write logic
protected val writeInData: UInt = Wire(UInt(theDataWidth.W))
protected val readOutData: UInt = Wire(UInt(theDataWidth.W))
// waitForRead: false, then ask for data; true, then data comes.
protected val waitForRead: Bool = RegInit(false.B) // use for letting increase slower when read
protected val nextValid: Bool = Wire(Bool())
protected val nextValidReg: Bool = RegInit(false.B) // one cycle after nextValid
protected val doReadWire: Bool = Wire(Bool())
protected val doWriteWire: Bool = Wire(Bool())
def readLogic(readOutDataIO: DecoupledIO[UInt], enable: Bool, idx: UInt, doDoneWire: Bool): Any = {
doReadWire := enable && readOutDataIO.ready
nextValid := doReadWire && !waitForRead
nextValidReg := nextValid
// when read, only enable signal becomes true and ready, then waitForRead signal began to flip
waitForRead := Mux(!doDoneWire && readOutDataIO.ready, Mux(doReadWire, !waitForRead, waitForRead), false.B)
readOutDataIO.valid := nextValidReg // reg next, so one cycle later, data will be read out with valid signal
readOutData := theSRAM.read(idx, nextValid)
readOutDataIO.bits := readOutData
}
def writeLogic(writeInDataIO: DecoupledIO[UInt], enable: Bool, idx: UInt): Any = {
doWriteWire := enable
writeInData := writeInDataIO.bits
writeInDataIO.ready := doWriteWire
when(writeInDataIO.fire()) {
theSRAM.write(idx, writeInData)
}
}
def debugLogic(debugIO: SRAMCommonDebugIO, enable: Bool): Any = {
debugIO.waitForRead := waitForRead
debugIO.doReadWire := enable
}
}
class InActSRAMCommon(private val theSRAMSize: Int, private val theDataWidth: Int, debug: Boolean)
extends SRAMCommon(theSRAMSize, theDataWidth) with GNMFCS2Config {
theSRAM.suggestName("oneInActSRAM")
val io: InACTSRAMCommonIO = IO(new InACTSRAMCommonIO(theSRAMSize, theDataWidth))
// use lookup table to store the start address in the SRAM of each stream
protected val adrLookUpTable: Mem[UInt] = Mem(2*inActStreamNum + 1, UInt(log2Ceil(theSRAMSize).W))
adrLookUpTable.suggestName("addressLookUpTableMem")
protected val noZero :: oneZero :: twoZeros :: Nil = Enum(3)
protected val zeroState: Seq[UInt] = Seq.fill(2){RegInit(noZero)} // 0 for read, 1 for write
zeroState.head.suggestName("readZeroStateReg")
zeroState.last.suggestName("writeZeroStateReg")
protected val meetZeroWire: Vec[Bool] = Wire(Vec(2, Bool())) // 0 for read, 1 for write
meetZeroWire.suggestName("meetZeroWire")
protected val writeDoneWire: Bool = Wire(Bool())
protected val readDoneWire: Bool = Wire(Bool())
protected val writeIdxReg: UInt = RegInit(0.U(width = log2Ceil(theSRAMSize).W)) // as counter
writeIdxReg.suggestName("writeIdx")
protected val writeIdxRegPlusOne: UInt = writeIdxReg + 1.U
protected val readIdxAccReg: UInt = RegInit(0.U(width = log2Ceil(theSRAMSize).W)) // as counter
readIdxAccReg.suggestName("readIdxAcc")
// lookup table
protected val lookupTableWriteIdx: UInt = RegInit(0.U(log2Ceil(2*inActStreamNum).W))
protected val lookupTableWriteIdxPlusOne: UInt = lookupTableWriteIdx + 1.U
protected val readStartIdx: UInt = adrLookUpTable.read(io.ctrlPath.readIO.adr)
protected val writeMeetZeroReg: Bool = zeroState.last === oneZero
when (writeMeetZeroReg) {
// when meet zero, we need to record the start index of new stream
adrLookUpTable.write(lookupTableWriteIdxPlusOne, writeIdxReg) // let index zero's value be zero all the time
}
lookupTableWriteIdx := Mux(writeDoneWire, 0.U,
Mux(writeMeetZeroReg && doWriteWire, lookupTableWriteIdxPlusOne, lookupTableWriteIdx)
) // it will increase at second write cycle, as reg next be true as default
// SRAM read write logic
// if meet two continuous zeros, then the group of data finishes, also write finish
protected val writeMeetTwoZerosWire: Bool = Wire(Bool())
writeMeetTwoZerosWire.suggestName("writeMeetTwoZeros")
io.ctrlPath.readIO.done := readDoneWire && zeroState.head =/= oneZero
io.ctrlPath.writeIO.done := writeDoneWire && writeIdxReg =/= 0.U
// write logic
writeLogic(io.dataPath.inIOs.data, enable = io.ctrlPath.writeIO.enable,
idx = writeIdxReg
)
// when data has been write in, then write index increase one
writeIdxReg := Mux(io.ctrlPath.writeIO.done, 0.U, Mux(doWriteWire, writeIdxRegPlusOne, writeIdxReg))
// read logic
readLogic(io.dataPath.outIOs.data, enable = io.ctrlPath.readIO.enable,
idx = readIdxAccReg + readStartIdx, doDoneWire = readDoneWire
)
// when data has been read out, then read index increase one
readIdxAccReg := Mux(readDoneWire || !io.ctrlPath.readIO.enable, 0.U,
Mux(nextValidReg, readIdxAccReg + 1.U, readIdxAccReg)
)
// do finish?
meetZeroWire.head := readOutData === 0.U
meetZeroWire.last := writeInData === 0.U
protected val canJump = Seq(io.dataPath.outIOs.data.fire(), io.dataPath.inIOs.data.fire())
for (i <- 0 until 2) {
switch(zeroState(i)) {
is (noZero) {
when (meetZeroWire(i) && canJump(i) ) {
zeroState(i) := oneZero
}
}
is (oneZero) {
if (i == 1) {
// when write, then just see current data.
when (meetZeroWire(i) && canJump(i) ) {
zeroState(i) := twoZeros
} .otherwise {
zeroState(i) := noZero
}
} else {
// While read, only waitForRead is true, then we can judge it.
when (waitForRead) {
when (meetZeroWire(i)) {
zeroState(i) := twoZeros
} .otherwise {
zeroState(i) := noZero
}
}
}
}
is (twoZeros) {
zeroState(i) := noZero // we assume there will not be three continuous zeros
}
}
}
readDoneWire := meetZeroWire.head && waitForRead && io.ctrlPath.readIO.enable// or meet one zero
/** inAct needs write twice, former and later */
protected val writeDoneOnce: Bool = RegInit(false.B)
writeDoneOnce.suggestName("writeDoneOnce")
writeDoneOnce := Mux(writeMeetTwoZerosWire, !writeDoneOnce, writeDoneOnce)
writeDoneWire := writeDoneOnce && writeMeetTwoZerosWire
writeMeetTwoZerosWire := zeroState.last === twoZeros // that's two zeros, only write need
// debug io
if (debug) {
debugLogic(io.debugIO, nextValidReg)
io.debugIO.currentData := Mux(io.ctrlPath.readIO.enable, readOutData, writeInData)
io.debugIO.meetOneZero := readDoneWire
io.debugIO.zeroState := Mux(io.ctrlPath.readIO.enable, zeroState.head, zeroState.last)
io.debugIO.idx := Mux(!io.ctrlPath.writeIO.enable, readIdxAccReg + readStartIdx, writeIdxReg)
io.debugIO.indexAcc := readIdxAccReg
io.debugIO.lookupIdx := lookupTableWriteIdx
io.debugIO.doWriteWire := doWriteWire
} else {
io.debugIO <> DontCare
}
}
class InActSRAMBank(debug: Boolean) extends Module with ClusterSRAMConfig {
val io: InActSRAMBankIO = IO(new InActSRAMBankIO)
protected val adrSRAM: InActSRAMCommon = Module(new InActSRAMCommon(inActAdrSRAMSize, inActAdrWidth, debug))
adrSRAM.suggestName("adrSRAM")
protected val dataSRAM: InActSRAMCommon = Module(new InActSRAMCommon(inActDataSRAMSize, inActDataWidth, debug))
dataSRAM.suggestName("dataSRAM")
protected val adrSRAMIO: InACTSRAMCommonIO = adrSRAM.io
protected val dataSRAMIO: InACTSRAMCommonIO = dataSRAM.io
protected val bothDoneWire: Vec[Bool] = Wire(Vec(2,Bool()))
protected val doneAtSameTimeWire: Vec[Bool] = Wire(Vec(2,Bool()))
protected val inActIdle :: inActDoing :: inActWaitAdr :: inActWaitData :: Nil = Enum(4)
protected val inActState: Seq[UInt] = Seq.fill(2){RegInit(inActIdle)}
inActState.head.suggestName("inActReadState")
inActState.last.suggestName("inActWriteState")
protected val currentDoingWire: Vec[Bool] = Wire(Vec(2, Bool()))
protected val adrDoneWire: Vec[Bool] = Wire(Vec(2, Bool()))
protected val dataDoneWire: Vec[Bool] = Wire(Vec(2, Bool()))
adrDoneWire.head := adrSRAMIO.ctrlPath.readIO.done
dataDoneWire.head := dataSRAMIO.ctrlPath.readIO.done
adrDoneWire.last := adrSRAMIO.ctrlPath.writeIO.done
dataDoneWire.last := dataSRAMIO.ctrlPath.writeIO.done
adrSRAMIO.ctrlPath.readIO.enable := currentDoingWire.head || (inActState.head === inActWaitAdr)
dataSRAMIO.ctrlPath.readIO.enable := currentDoingWire.head || (inActState.head === inActWaitData)
adrSRAMIO.ctrlPath.writeIO.enable := currentDoingWire.last || (inActState.last === inActWaitAdr)
dataSRAMIO.ctrlPath.writeIO.enable := currentDoingWire.last || (inActState.last === inActWaitData)
protected val topCtrlSeq = Seq(io.ctrlPath.readIO, io.ctrlPath.writeIO)
for (i <- 0 until 2) {
currentDoingWire(i) := inActState(i) === inActDoing
doneAtSameTimeWire(i) := adrDoneWire(i) && dataDoneWire(i)
bothDoneWire(i) := ((inActState(i) === inActWaitData) && dataDoneWire(i)) ||
((inActState(i) === inActWaitAdr) && adrDoneWire(i)) ||
(currentDoingWire(i) && doneAtSameTimeWire(i))
switch(inActState(i)) {
is (inActIdle) {
when (topCtrlSeq(i).enable) {
inActState(i) := inActDoing
}
}
is (inActDoing) {
when (doneAtSameTimeWire(i)) {
inActState(i) := inActIdle
} .elsewhen (adrDoneWire(i)) {
inActState(i) := inActWaitData
} .elsewhen (dataDoneWire(i)) {
inActState(i) := inActWaitAdr
} .otherwise {
inActState(i) := inActDoing
}
}
is (inActWaitData) {
when (dataDoneWire(i)) {
inActState(i) := inActIdle
}
}
is (inActWaitAdr) {
when (adrDoneWire(i)) {
inActState(i) := inActIdle
}
}
}
}
// SRAM connections
adrSRAMIO.dataPath.inIOs <> io.dataPath.inIOs.adrIOs
adrSRAMIO.dataPath.outIOs <> io.dataPath.outIOs.adrIOs
dataSRAMIO.dataPath.inIOs <> io.dataPath.inIOs.dataIOs
dataSRAMIO.dataPath.outIOs <> io.dataPath.outIOs.dataIOs
// control path
Seq(adrSRAMIO.ctrlPath, dataSRAMIO.ctrlPath).foreach({ x =>
x.readIO.adr := io.ctrlPath.readIO.adr // address for lookup table
x.writeIO.adr := DontCare
})
io.ctrlPath.readIO.done := bothDoneWire.head
io.ctrlPath.writeIO.done := bothDoneWire.last
if (debug) {
io.debugIO.theState := Mux(!io.ctrlPath.writeIO.enable, inActState.head, inActState.last)
Seq(io.debugIO.adrDebug, io.debugIO.dataDebug).zip(Seq(adrSRAMIO, dataSRAMIO)).foreach({case (topIO, sram) =>
topIO.subDone := Mux(!io.ctrlPath.writeIO.enable, sram.ctrlPath.readIO.done, sram.ctrlPath.writeIO.done)
topIO.commonDebug <> sram.debugIO
})
} else {
io.debugIO <> DontCare
}
}
class SRAMCommonCtrlIO(private val theMemSize: Int) extends Bundle {
val writeIO = new MeMReWrIO(theMemSize)
val readIO = new MeMReWrIO(theMemSize)
}
class MeMReWrIO(private val theMemSize: Int) extends Bundle {
val enable: Bool = Input(Bool())
val adr: UInt = Input(UInt(log2Ceil(theMemSize).W))
val done: Bool = Output(Bool())
}
class SRAMCommonDebugIO(private val theSRAMSize: Int, private val theDataWidth: Int) extends Bundle {
val waitForRead: Bool = Output(Bool())
val doReadWire: Bool = Output(Bool())
}
trait InActSpecialDebugIO extends Bundle with ClusterSRAMConfig {
val indexAcc: UInt = Output(UInt(30.W))
val idx: UInt = Output(UInt(log2Ceil(inActDataSRAMSize).W))
val currentData: UInt = Output(UInt(inActDataWidth.W))
val meetOneZero: Bool = Output(Bool())
val zeroState: UInt = Output(UInt(2.W))
val lookupIdx: UInt = Output(UInt(20.W))
val doWriteWire: Bool = Output(Bool())
}
class InActSRAMBankDebugSubIO(private val theSRAMSize: Int, private val theDataWidth: Int) extends Bundle {
val commonDebug = new SRAMCommonDebugIO(theSRAMSize, theDataWidth) with InActSpecialDebugIO
val subDone: Bool = Output(Bool())
}
class InACTSRAMCommonIO(private val theSRAMSize: Int, private val theDataWidth: Int) extends Bundle {
val dataPath = new StreamBitsInOutIO(theDataWidth)
val ctrlPath = new SRAMCommonCtrlIO(theMemSize = theSRAMSize)
val debugIO = new SRAMCommonDebugIO(theSRAMSize, theDataWidth) with InActSpecialDebugIO
}
class InActSRAMBankDebugIO extends Bundle with ClusterSRAMConfig {
val theState: UInt = Output(UInt(2.W))
val adrDebug = new InActSRAMBankDebugSubIO(inActAdrSRAMSize, inActAdrWidth)
val dataDebug = new InActSRAMBankDebugSubIO(inActDataSRAMSize, inActDataWidth)
}
class InActSRAMBankIO extends Bundle with ClusterSRAMConfig with GNMFCS2Config {
val dataPath = new CSCStreamInOutIO(inActAdrWidth, inActDataWidth)
// address data width only need to index stream numbers
// as it need the stream number to calculate the start index of the CSC format data
val ctrlPath = new SRAMCommonCtrlIO(inActStreamNum)
val debugIO = new InActSRAMBankDebugIO
}
class StreamBitsInOutIO(private val theDataWidth: Int) extends Bundle {
val inIOs: StreamBitsIO = Flipped(new StreamBitsIO(theDataWidth))
val outIOs = new StreamBitsIO(theDataWidth)
}
class CSCStreamInOutIO(private val adrWidth: Int, private val dataWidth: Int) extends Bundle {
val inIOs: CSCStreamIO = Flipped(new CSCStreamIO(adrWidth = adrWidth, dataWidth = dataWidth))
val outIOs = new CSCStreamIO(adrWidth = adrWidth, dataWidth = dataWidth)
}
class PSumSRAMBankIO extends Bundle with ClusterSRAMConfig {
val dataPath = new PSumSRAMDataIO(psDataWidth)
val ctrlPath = new SRAMCommonCtrlIO(theMemSize = pSumSRAMSize)
val debugIO = new SRAMCommonDebugIO(pSumSRAMSize, psDataWidth)
}
class PSumSRAMDataIO(private val dataWidth: Int) extends Bundle {
val inIOs: DecoupledIO[UInt] = Flipped(Decoupled(UInt(dataWidth.W)))
val outIOs: DecoupledIO[UInt] = Decoupled(UInt(dataWidth.W))
}
class GLBClusterIO extends Bundle {
val dataPath = new GLBClusterDataIO
val ctrlPath = new GLBClusterCtrlIO
val debugIO = new GLBClusterDebugIO
}
class WeightGLBIO extends Bundle with ClusterSRAMConfig {
val dataPath = new CSCStreamInOutIO(weightAdrWidth, weightDataWidth)
}
class GLBClusterCtrlIO extends Bundle with GNMFCS1Config with ClusterSRAMConfig {
val inActIO: Vec[SRAMCommonCtrlIO] = Vec(inActSRAMNum, new SRAMCommonCtrlIO(inActDataSRAMSize))
val pSumIO: Vec[SRAMCommonCtrlIO] = Vec(pSumSRAMNum, new SRAMCommonCtrlIO(pSumSRAMSize))
}
class GLBClusterDataIO extends Bundle with ClusterSRAMConfig {
val inActIO: Vec[CSCStreamInOutIO] = Vec(inActSRAMNum, new CSCStreamInOutIO(inActAdrWidth, inActDataWidth))
val weightIO: Vec[CSCStreamInOutIO] = Vec(weightRouterNum, new CSCStreamInOutIO(weightAdrWidth, weightDataWidth))
val pSumIO: Vec[PSumSRAMDataIO] = Vec(pSumSRAMNum, new PSumSRAMDataIO(psDataWidth))
}
class GLBClusterDebugIO extends Bundle with ClusterSRAMConfig {
val oneInActSRAMDone: Vec[Bool] = Output(Vec(inActSRAMNum, Bool()))
val onePSumSRAMDone: Vec[Bool] = Output(Vec(pSumSRAMNum, Bool()))
val inActDebugIO: Vec[InActSRAMBankDebugIO] = Vec(inActSRAMNum, new InActSRAMBankDebugIO)
val pSumDebugIO: Vec[SRAMCommonDebugIO] = Vec(pSumSRAMNum, new SRAMCommonDebugIO(pSumSRAMSize, psDataWidth))
}