2016-06-28 20:21:38 +02:00
|
|
|
package uncore.devices
|
2016-04-27 09:15:00 +02:00
|
|
|
|
|
|
|
import Chisel._
|
|
|
|
import junctions._
|
2016-06-28 20:21:38 +02:00
|
|
|
import uncore.tilelink._
|
|
|
|
import uncore.util._
|
2016-04-27 09:15:00 +02:00
|
|
|
import cde.{Parameters, Field}
|
|
|
|
|
|
|
|
class ROMSlave(contents: Seq[Byte])(implicit val p: Parameters) extends Module
|
|
|
|
with HasTileLinkParameters
|
|
|
|
with HasAddrMapParameters {
|
|
|
|
val io = new ClientUncachedTileLinkIO().flip
|
|
|
|
|
|
|
|
val acq = Queue(io.acquire, 1)
|
2016-05-01 02:34:12 +02:00
|
|
|
val single_beat = acq.bits.isBuiltInType(Acquire.getType)
|
|
|
|
val multi_beat = acq.bits.isBuiltInType(Acquire.getBlockType)
|
|
|
|
assert(!acq.valid || single_beat || multi_beat, "unsupported ROMSlave operation")
|
|
|
|
|
|
|
|
val addr_beat = Reg(UInt())
|
|
|
|
when (io.grant.fire()) { addr_beat := addr_beat + UInt(1) }
|
|
|
|
when (io.acquire.fire()) { addr_beat := io.acquire.bits.addr_beat }
|
2016-04-27 09:15:00 +02:00
|
|
|
|
|
|
|
val byteWidth = tlDataBits / 8
|
2016-06-02 06:56:24 +02:00
|
|
|
val rows = (contents.size + byteWidth - 1)/byteWidth
|
2016-04-27 09:15:00 +02:00
|
|
|
val rom = Vec.tabulate(rows) { i =>
|
|
|
|
val slice = contents.slice(i*byteWidth, (i+1)*byteWidth)
|
|
|
|
UInt(slice.foldRight(BigInt(0)) { case (x,y) => (y << 8) + (x.toInt & 0xFF) }, byteWidth*8)
|
|
|
|
}
|
2016-05-01 02:34:12 +02:00
|
|
|
val raddr = Cat(acq.bits.addr_block, addr_beat)
|
|
|
|
val rdata = rom(if (rows == 1) UInt(0) else raddr(log2Up(rom.size)-1,0))
|
2016-04-27 09:15:00 +02:00
|
|
|
|
2016-05-01 02:34:12 +02:00
|
|
|
val last = !multi_beat || addr_beat === UInt(tlDataBeats-1)
|
2016-04-27 09:15:00 +02:00
|
|
|
io.grant.valid := acq.valid
|
2016-05-01 02:34:12 +02:00
|
|
|
acq.ready := io.grant.ready && last
|
2016-04-27 09:15:00 +02:00
|
|
|
io.grant.bits := Grant(
|
|
|
|
is_builtin_type = Bool(true),
|
|
|
|
g_type = acq.bits.getBuiltInGrantType(),
|
|
|
|
client_xact_id = acq.bits.client_xact_id,
|
|
|
|
manager_xact_id = UInt(0),
|
2016-05-01 02:34:12 +02:00
|
|
|
addr_beat = addr_beat,
|
2016-04-27 09:15:00 +02:00
|
|
|
data = rdata)
|
|
|
|
}
|
2016-05-02 22:58:41 +02:00
|
|
|
|
|
|
|
class NastiROM(contents: Seq[Byte])(implicit p: Parameters) extends Module {
|
|
|
|
val io = new NastiIO().flip
|
|
|
|
val ar = Queue(io.ar, 1)
|
|
|
|
|
|
|
|
// This assumes ROMs are in read-only parts of the address map.
|
|
|
|
// Reuse b_queue code from NastiErrorSlave if this assumption is bad.
|
|
|
|
when (ar.valid) { assert(ar.bits.len === UInt(0), "Can't burst-read from NastiROM") }
|
|
|
|
assert(!(io.aw.valid || io.w.valid), "Can't write to NastiROM")
|
|
|
|
io.aw.ready := Bool(false)
|
|
|
|
io.w.ready := Bool(false)
|
|
|
|
io.b.valid := Bool(false)
|
|
|
|
|
|
|
|
val byteWidth = io.r.bits.nastiXDataBits / 8
|
2016-06-02 06:56:24 +02:00
|
|
|
val rows = (contents.size + byteWidth - 1)/byteWidth
|
2016-05-02 22:58:41 +02:00
|
|
|
val rom = Vec.tabulate(rows) { i =>
|
|
|
|
val slice = contents.slice(i*byteWidth, (i+1)*byteWidth)
|
|
|
|
UInt(slice.foldRight(BigInt(0)) { case (x,y) => (y << 8) + (x.toInt & 0xFF) }, byteWidth*8)
|
|
|
|
}
|
2016-08-09 23:39:06 +02:00
|
|
|
val rdata = rom(if (rows == 1) UInt(0) else ar.bits.addr(log2Up(contents.size)-1,log2Up(byteWidth)))
|
2016-05-02 22:58:41 +02:00
|
|
|
|
|
|
|
io.r <> ar
|
|
|
|
io.r.bits := NastiReadDataChannel(ar.bits.id, rdata)
|
|
|
|
}
|