|  | // Copyright 2023 Google LLC | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | package common | 
|  |  | 
|  | import chisel3._ | 
|  | import chisel3.util._ | 
|  | import _root_.circt.stage.ChiselStage | 
|  |  | 
|  | // FifoX with entry output and no output registration stage. | 
|  |  | 
|  | object FifoXe { | 
|  | def apply[T <: Data](t: T, x: Int, n: Int) = { | 
|  | Module(new FifoXe(t, x, n)) | 
|  | } | 
|  | } | 
|  |  | 
|  | class FifoXe[T <: Data](t: T, x:Int, n: Int) extends Module { | 
|  | val io = IO(new Bundle { | 
|  | val in  = Flipped(Decoupled(Vec(x, Valid(t)))) | 
|  | val out = Decoupled(t) | 
|  | val count = Output(UInt(log2Ceil(n+1).W)) | 
|  | val entry = Output(Vec(n, Valid(t))) | 
|  | val nempty = Output(Bool()) | 
|  | }) | 
|  |  | 
|  | def Increment(a: UInt, b: UInt): UInt = { | 
|  | val c = a +& b | 
|  | val d = Mux(c < n.U, c, c - n.U)(a.getWidth - 1, 0) | 
|  | d | 
|  | } | 
|  |  | 
|  | val mem = Mem(n, t) | 
|  |  | 
|  | val inxpos = RegInit(VecInit((0 until x).map(x => x.U((log2Ceil(n) + 1).W)))) | 
|  | val outpos = RegInit(0.U(log2Ceil(n).W)) | 
|  | val mcount = RegInit(0.U(log2Ceil(n+1).W)) | 
|  | val nempty = RegInit(false.B) | 
|  |  | 
|  | io.count := mcount | 
|  | io.nempty := nempty | 
|  |  | 
|  | val ivalid = io.in.valid && io.in.ready | 
|  | val ovalid = io.out.valid && io.out.ready | 
|  |  | 
|  | val iactive = Cat((0 until x).reverse.map(x => io.in.bits(x).valid)) | 
|  |  | 
|  | val icount = PopCount(iactive) | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  | // Fifo Control. | 
|  | when (ivalid) { | 
|  | for (i <- 0 until x) { | 
|  | inxpos(i) := Increment(inxpos(i), icount) | 
|  | } | 
|  | } | 
|  |  | 
|  | when (ovalid) { | 
|  | outpos := Increment(outpos, 1.U) | 
|  | } | 
|  |  | 
|  | val inc = MuxOR(ivalid, icount) | 
|  | val dec = ovalid | 
|  |  | 
|  | when (ivalid || ovalid) { | 
|  | val nxtcount = mcount + inc - dec | 
|  | mcount := nxtcount | 
|  | nempty := nxtcount =/= 0.U | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  | // Fifo Input. | 
|  | val inxvalid = FifoXValid(iactive) | 
|  |  | 
|  | for (i <- 0 until n) { | 
|  | val valid = Cat( | 
|  | (0 until x).reverse.map(q => | 
|  | if (q == 0) { inxpos(0) === i.U && inxvalid(0)(0) } else { | 
|  | (0 to q).map(y => | 
|  | inxpos(y) === i.U && inxvalid(y)(q) | 
|  | ).reduce(_ || _) | 
|  | } | 
|  | ) | 
|  | ) | 
|  |  | 
|  | when (ivalid) { | 
|  | when (PopCount(valid) >= 1.U) { | 
|  | val idx = PriorityEncoder(valid) | 
|  | mem(i) := io.in.bits(idx).bits | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  | // Valid Entries. | 
|  | val active = RegInit(0.U(n.W)) | 
|  |  | 
|  | val activeSet = MuxOR(ivalid, | 
|  | (0 until x).map(i => (icount >= (i + 1).U) << inxpos(i)).reduce(_ | _) | 
|  | ) | 
|  |  | 
|  | val activeClr = MuxOR(io.out.valid && io.out.ready, 1.U << outpos) | 
|  |  | 
|  | when (io.in.valid && io.in.ready || io.out.valid && io.out.ready) { | 
|  | active := (active | activeSet) & ~activeClr | 
|  | } | 
|  |  | 
|  | // --------------------------------------------------------------------------- | 
|  | // Interface. | 
|  | io.in.ready := mcount <= (n.U - icount) | 
|  |  | 
|  | io.out.valid := mcount =/= 0.U | 
|  | io.out.bits := mem(outpos) | 
|  |  | 
|  | assert(mcount <= n.U) | 
|  |  | 
|  | for (i <- 0 until n) { | 
|  | io.entry(i).valid := active(i) | 
|  | io.entry(i).bits := mem(i) | 
|  | } | 
|  | } | 
|  |  | 
|  | object EmitFifoXe extends App { | 
|  | ChiselStage.emitSystemVerilogFile(new FifoXe(UInt(8.W), 4, 10), args) | 
|  | } |