2016-08-27 00:48:48 +02:00
|
|
|
// See LICENSE for license details.
|
|
|
|
|
|
|
|
package uncore.tilelink2
|
|
|
|
|
|
|
|
import Chisel._
|
|
|
|
|
2016-09-16 00:52:50 +02:00
|
|
|
class TLRegisterNode(address: AddressSet, concurrency: Option[Int] = None, beatBytes: Int = 4, undefZero: Boolean = true)
|
2016-08-27 00:48:48 +02:00
|
|
|
extends TLManagerNode(beatBytes, TLManagerParameters(
|
|
|
|
address = Seq(address),
|
|
|
|
supportsGet = TransferSizes(1, beatBytes),
|
|
|
|
supportsPutPartial = TransferSizes(1, beatBytes),
|
|
|
|
supportsPutFull = TransferSizes(1, beatBytes),
|
|
|
|
fifoId = Some(0))) // requests are handled in order
|
|
|
|
{
|
|
|
|
require (!address.strided)
|
|
|
|
|
|
|
|
// Calling this method causes the matching TL2 bundle to be
|
|
|
|
// configured to route all requests to the listed RegFields.
|
|
|
|
def regmap(mapping: RegField.Map*) = {
|
2016-08-30 00:33:10 +02:00
|
|
|
val a = bundleIn(0).a
|
2016-08-27 00:48:48 +02:00
|
|
|
val d = bundleIn(0).d
|
2016-08-30 00:33:10 +02:00
|
|
|
val edge = edgesIn(0)
|
2016-09-07 08:46:44 +02:00
|
|
|
|
|
|
|
// Please forgive me ...
|
|
|
|
val baseEnd = 0
|
|
|
|
val (sizeEnd, sizeOff) = (edge.bundle.sizeBits + baseEnd, baseEnd)
|
|
|
|
val (sourceEnd, sourceOff) = (edge.bundle.sourceBits + sizeEnd, sizeEnd)
|
|
|
|
val (addrLoEnd, addrLoOff) = (log2Ceil(beatBytes) + sourceEnd, sourceEnd)
|
|
|
|
|
|
|
|
val params = RegMapperParams(log2Up(address.mask+1), beatBytes, addrLoEnd)
|
2016-08-31 22:37:20 +02:00
|
|
|
val in = Wire(Decoupled(new RegMapperInput(params)))
|
2016-08-30 00:33:10 +02:00
|
|
|
in.bits.read := a.bits.opcode === TLMessages.Get
|
2016-09-07 08:46:44 +02:00
|
|
|
in.bits.index := a.bits.addr_hi
|
2016-08-30 00:33:10 +02:00
|
|
|
in.bits.data := a.bits.data
|
2016-08-31 00:06:37 +02:00
|
|
|
in.bits.mask := a.bits.mask
|
2016-09-07 08:46:44 +02:00
|
|
|
in.bits.extra := Cat(edge.addr_lo(a.bits), a.bits.source, a.bits.size)
|
2016-08-30 00:33:10 +02:00
|
|
|
|
|
|
|
// Invoke the register map builder
|
2016-09-16 00:52:50 +02:00
|
|
|
val (endIndex, out) = RegMapper(beatBytes, concurrency, undefZero, in, mapping:_*)
|
2016-08-27 00:48:48 +02:00
|
|
|
|
2016-08-30 00:33:10 +02:00
|
|
|
// All registers must fit inside the device address space
|
|
|
|
require (address.mask >= (endIndex-1)*beatBytes)
|
2016-08-27 00:48:48 +02:00
|
|
|
|
2016-08-30 00:33:10 +02:00
|
|
|
// No flow control needed
|
|
|
|
in.valid := a.valid
|
|
|
|
a.ready := in.ready
|
|
|
|
d.valid := out.valid
|
|
|
|
out.ready := d.ready
|
2016-08-27 00:48:48 +02:00
|
|
|
|
2016-09-07 08:46:44 +02:00
|
|
|
// We must restore the size and addr_lo to enable width adapters to work
|
|
|
|
d.bits := edge.AccessAck(
|
|
|
|
fromAddress = out.bits.extra(addrLoEnd-1, addrLoOff),
|
|
|
|
fromSink = UInt(0), // our unique sink id
|
|
|
|
toSource = out.bits.extra(sourceEnd-1, sourceOff),
|
|
|
|
lgSize = out.bits.extra(sizeEnd-1, sizeOff))
|
|
|
|
|
2016-08-27 00:48:48 +02:00
|
|
|
// avoid a Mux on the data bus by manually overriding two fields
|
2016-08-30 00:33:10 +02:00
|
|
|
d.bits.data := out.bits.data
|
|
|
|
d.bits.opcode := Mux(out.bits.read, TLMessages.AccessAckData, TLMessages.AccessAck)
|
2016-09-05 01:47:18 +02:00
|
|
|
|
|
|
|
// Tie off unused channels
|
|
|
|
bundleIn(0).b.valid := Bool(false)
|
|
|
|
bundleIn(0).c.ready := Bool(true)
|
|
|
|
bundleIn(0).e.ready := Bool(true)
|
2016-08-27 00:48:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
object TLRegisterNode
|
|
|
|
{
|
2016-09-16 00:52:50 +02:00
|
|
|
def apply(address: AddressSet, concurrency: Option[Int] = None, beatBytes: Int = 4, undefZero: Boolean = true) =
|
|
|
|
new TLRegisterNode(address, concurrency, beatBytes, undefZero)
|
2016-08-27 00:48:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// These convenience methods below combine to make it possible to create a TL2
|
|
|
|
// register mapped device from a totally abstract register mapped device.
|
|
|
|
// See GPIO.scala in this directory for an example
|
|
|
|
|
2016-09-16 00:52:50 +02:00
|
|
|
abstract class TLRegisterRouterBase(address: AddressSet, interrupts: Int, concurrency: Option[Int], beatBytes: Int, undefZero: Boolean) extends LazyModule
|
2016-08-27 00:48:48 +02:00
|
|
|
{
|
2016-09-16 00:52:50 +02:00
|
|
|
val node = TLRegisterNode(address, concurrency, beatBytes, undefZero)
|
2016-09-09 00:17:30 +02:00
|
|
|
val intnode = IntSourceNode(name + s" @ ${address.base}", interrupts)
|
2016-08-27 00:48:48 +02:00
|
|
|
}
|
|
|
|
|
2016-09-09 00:17:30 +02:00
|
|
|
case class TLRegBundleArg(interrupts: Vec[Vec[Bool]], in: Vec[TLBundle])
|
2016-08-27 00:48:48 +02:00
|
|
|
|
2016-09-09 00:17:30 +02:00
|
|
|
class TLRegBundleBase(arg: TLRegBundleArg) extends Bundle
|
|
|
|
{
|
|
|
|
val interrupts = arg.interrupts
|
|
|
|
val in = arg.in
|
|
|
|
}
|
|
|
|
|
|
|
|
class TLRegBundle[P](val params: P, arg: TLRegBundleArg) extends TLRegBundleBase(arg)
|
|
|
|
|
|
|
|
class TLRegModule[P, B <: TLRegBundleBase](val params: P, bundleBuilder: => B, router: TLRegisterRouterBase)
|
2016-08-31 19:25:46 +02:00
|
|
|
extends LazyModuleImp(router) with HasRegMap
|
2016-08-27 00:48:48 +02:00
|
|
|
{
|
2016-08-29 20:08:37 +02:00
|
|
|
val io = bundleBuilder
|
2016-09-09 00:17:30 +02:00
|
|
|
val interrupts = if (io.interrupts.isEmpty) Vec(0, Bool()) else io.interrupts(0)
|
2016-08-31 19:25:46 +02:00
|
|
|
def regmap(mapping: RegField.Map*) = router.node.regmap(mapping:_*)
|
2016-08-27 00:48:48 +02:00
|
|
|
}
|
|
|
|
|
2016-09-09 00:17:30 +02:00
|
|
|
class TLRegisterRouter[B <: TLRegBundleBase, M <: LazyModuleImp]
|
2016-09-16 00:52:50 +02:00
|
|
|
(val base: BigInt, val interrupts: Int = 0, val size: BigInt = 4096, val concurrency: Option[Int] = None, val beatBytes: Int = 4, undefZero: Boolean = true)
|
2016-09-09 00:17:30 +02:00
|
|
|
(bundleBuilder: TLRegBundleArg => B)
|
2016-08-31 19:25:46 +02:00
|
|
|
(moduleBuilder: (=> B, TLRegisterRouterBase) => M)
|
2016-09-16 00:52:50 +02:00
|
|
|
extends TLRegisterRouterBase(AddressSet(base, size-1), interrupts, concurrency, beatBytes, undefZero)
|
2016-08-27 00:48:48 +02:00
|
|
|
{
|
|
|
|
require (isPow2(size))
|
2016-09-12 00:43:04 +02:00
|
|
|
// require (size >= 4096) ... not absolutely required, but highly recommended
|
2016-08-27 00:48:48 +02:00
|
|
|
|
2016-09-09 00:17:30 +02:00
|
|
|
lazy val module = moduleBuilder(bundleBuilder(TLRegBundleArg(intnode.bundleOut, node.bundleIn)), this)
|
2016-08-27 00:48:48 +02:00
|
|
|
}
|