[firechip] Rework FireSim clocking to be more similar to default CY targets
This commit is contained in:
@@ -68,9 +68,7 @@ class WithFireSimIOCellModels extends Config((site, here, up) => {
|
|||||||
class WithSerialBridge extends OverrideHarnessBinder({
|
class WithSerialBridge extends OverrideHarnessBinder({
|
||||||
(system: CanHavePeripherySerial, th: HasHarnessSignalReferences, ports: Seq[ClockedIO[SerialIO]]) => {
|
(system: CanHavePeripherySerial, th: HasHarnessSignalReferences, ports: Seq[ClockedIO[SerialIO]]) => {
|
||||||
ports.map { p =>
|
ports.map { p =>
|
||||||
withClockAndReset(p.clock, th.harnessReset) {
|
SerialBridge(p.clock, p.bits, MainMemoryConsts.globalName)(GetSystemParameters(system))
|
||||||
SerialBridge(p.clock, p.bits, MainMemoryConsts.globalName)(GetSystemParameters(system))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
Nil
|
Nil
|
||||||
}
|
}
|
||||||
@@ -79,7 +77,7 @@ class WithSerialBridge extends OverrideHarnessBinder({
|
|||||||
class WithNICBridge extends OverrideHarnessBinder({
|
class WithNICBridge extends OverrideHarnessBinder({
|
||||||
(system: CanHavePeripheryIceNIC, th: HasHarnessSignalReferences, ports: Seq[ClockedIO[NICIOvonly]]) => {
|
(system: CanHavePeripheryIceNIC, th: HasHarnessSignalReferences, ports: Seq[ClockedIO[NICIOvonly]]) => {
|
||||||
val p: Parameters = GetSystemParameters(system)
|
val p: Parameters = GetSystemParameters(system)
|
||||||
ports.map { n => withClockAndReset(n.clock, th.harnessReset) { NICBridge(n.clock, n.bits)(p) } }
|
ports.map { n => NICBridge(n.clock, n.bits)(p) }
|
||||||
Nil
|
Nil
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -119,11 +117,7 @@ class WithFASEDBridge extends OverrideHarnessBinder({
|
|||||||
|
|
||||||
class WithTracerVBridge extends ComposeHarnessBinder({
|
class WithTracerVBridge extends ComposeHarnessBinder({
|
||||||
(system: CanHaveTraceIOModuleImp, th: HasHarnessSignalReferences, ports: Seq[TraceOutputTop]) => {
|
(system: CanHaveTraceIOModuleImp, th: HasHarnessSignalReferences, ports: Seq[TraceOutputTop]) => {
|
||||||
ports.map { p =>
|
ports.map { p => p.traces.map(tileTrace => TracerVBridge(tileTrace)(system.p)) }
|
||||||
p.traces.map(
|
|
||||||
tileTrace => withClockAndReset(tileTrace.clock, tileTrace.reset) { TracerVBridge(tileTrace)(system.p) }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
Nil
|
Nil
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -8,14 +8,15 @@ import chisel3.experimental.{IO}
|
|||||||
import freechips.rocketchip.prci._
|
import freechips.rocketchip.prci._
|
||||||
import freechips.rocketchip.subsystem.{BaseSubsystem, SubsystemDriveAsyncClockGroupsKey}
|
import freechips.rocketchip.subsystem.{BaseSubsystem, SubsystemDriveAsyncClockGroupsKey}
|
||||||
import freechips.rocketchip.config.{Field, Config, Parameters}
|
import freechips.rocketchip.config.{Field, Config, Parameters}
|
||||||
import freechips.rocketchip.diplomacy.{LazyModule, InModuleBody}
|
import freechips.rocketchip.diplomacy.{LazyModule, InModuleBody, ValName}
|
||||||
import freechips.rocketchip.util.{ResetCatchAndSync}
|
import freechips.rocketchip.util.{ResetCatchAndSync, RecordMap}
|
||||||
|
|
||||||
import midas.widgets.{Bridge, PeekPokeBridge, RationalClockBridge, RationalClock}
|
import midas.widgets.{Bridge, PeekPokeBridge, RationalClockBridge, RationalClock}
|
||||||
|
|
||||||
import chipyard._
|
import chipyard._
|
||||||
import chipyard.harness._
|
import chipyard.harness._
|
||||||
import chipyard.iobinders._
|
import chipyard.iobinders._
|
||||||
|
import chipyard.clocking.{FrequencyUtils, ClockGroupNamePrefixer, ClockGroupFrequencySpecifier, SimplePllConfiguration}
|
||||||
|
|
||||||
// Determines the number of times to instantiate the DUT in the harness.
|
// Determines the number of times to instantiate the DUT in the harness.
|
||||||
// Subsumes legacy supernode support
|
// Subsumes legacy supernode support
|
||||||
@@ -25,16 +26,6 @@ class WithNumNodes(n: Int) extends Config((pname, site, here) => {
|
|||||||
case NumNodes => n
|
case NumNodes => n
|
||||||
})
|
})
|
||||||
|
|
||||||
// Note, the main prerequisite for supporting an additional clock domain in a
|
|
||||||
// FireSim simulation is to supply an additional clock parameter
|
|
||||||
// (RationalClock) to the clock bridge (RationalClockBridge). The bridge
|
|
||||||
// produces a vector of clocks, based on the provided parameter list, which you
|
|
||||||
// may use freely without further modifications to your target design.
|
|
||||||
case class FireSimClockParameters(additionalClocks: Seq[RationalClock]) {
|
|
||||||
def numClocks(): Int = additionalClocks.size + 1
|
|
||||||
}
|
|
||||||
case object FireSimClockKey extends Field[FireSimClockParameters](FireSimClockParameters(Seq()))
|
|
||||||
|
|
||||||
// Hacky: Set before each node is generated. Ideally we'd give IO binders
|
// Hacky: Set before each node is generated. Ideally we'd give IO binders
|
||||||
// accesses to the the Harness's parameters instance. We could then alter that.
|
// accesses to the the Harness's parameters instance. We could then alter that.
|
||||||
object NodeIdx {
|
object NodeIdx {
|
||||||
@@ -43,107 +34,114 @@ object NodeIdx {
|
|||||||
def apply(): Int = idx
|
def apply(): Int = idx
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Under FireSim's current multiclock implementation there can be only a
|
||||||
|
* single clock bridge. This requires, therefore, that it be instantiated in
|
||||||
|
* the harness and reused across all supernode instances. This class attempts to
|
||||||
|
* memoize its instantiation such that it can be referenced from within a ClockScheme function.
|
||||||
|
*/
|
||||||
|
class ClockBridgeInstantiator {
|
||||||
|
private var _clockRecord: Option[RecordMap[Clock]] = None
|
||||||
|
|
||||||
|
def getClockRecord: RecordMap[Clock] = _clockRecord.get
|
||||||
|
|
||||||
|
def getClockRecordOrInstantiate(allClocks: Seq[RationalClock], baseClockName: String): RecordMap[Clock] = {
|
||||||
|
if (_clockRecord.isEmpty) {
|
||||||
|
require(allClocks.exists(_.name == baseClockName),
|
||||||
|
s"Provided base-clock name, ${baseClockName}, does not match a defined clock. Available clocks:\n " +
|
||||||
|
allClocks.map(_.name).mkString("\n "))
|
||||||
|
|
||||||
|
val baseClock = allClocks.find(_.name == baseClockName).get
|
||||||
|
val simplified = allClocks.map { c =>
|
||||||
|
c.copy(multiplier = c.multiplier * baseClock.divisor, divisor = c.divisor * baseClock.multiplier)
|
||||||
|
.simplify
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes clocks that have the same frequency before instantiating the
|
||||||
|
* clock bridge to avoid unnecessary BUFGCE use.
|
||||||
|
*/
|
||||||
|
val distinct = simplified.foldLeft(Seq(RationalClock(baseClockName, 1, 1))) { case (list, candidate) =>
|
||||||
|
if (list.exists { clock => clock.equalFrequency(candidate) }) list else list :+ candidate
|
||||||
|
}
|
||||||
|
|
||||||
|
val clockBridge = Module(new RationalClockBridge(distinct))
|
||||||
|
val cbVecTuples = distinct.zip(clockBridge.io.clocks)
|
||||||
|
val outputWire = Wire(RecordMap(allClocks.map { c => (c.name, Clock()) }:_*))
|
||||||
|
for (parameter <- allClocks) {
|
||||||
|
val (_, cbClockField) = cbVecTuples.find(_._1.equalFrequency(parameter)).get
|
||||||
|
outputWire(parameter.name).get := cbClockField
|
||||||
|
}
|
||||||
|
_clockRecord = Some(outputWire)
|
||||||
|
}
|
||||||
|
getClockRecord
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
case object ClockBridgeInstantiatorKey extends Field[ClockBridgeInstantiator](new ClockBridgeInstantiator)
|
||||||
|
case object FireSimBaseClockNameKey extends Field[String]("implicit_clock")
|
||||||
|
|
||||||
class WithFireSimSimpleClocks extends Config((site, here, up) => {
|
class WithFireSimSimpleClocks extends Config((site, here, up) => {
|
||||||
case ClockingSchemeKey => { chiptop: ChipTop =>
|
case ClockingSchemeKey => { chiptop: ChipTop =>
|
||||||
implicit val p = chiptop.p
|
implicit val p = chiptop.p
|
||||||
|
// Figure out what provides this in the chipyard scheme
|
||||||
|
implicit val valName = ValName("FireSimClocking")
|
||||||
|
|
||||||
val implicitClockSourceNode = ClockSourceNode(Seq(ClockSourceParameters()))
|
// Requires existence of undriven asyncClockGroups in subsystem
|
||||||
chiptop.implicitClockSinkNode := implicitClockSourceNode
|
val systemAsyncClockGroup = chiptop.lazySystem match {
|
||||||
|
case l: BaseSubsystem if (p(SubsystemDriveAsyncClockGroupsKey).isEmpty) =>
|
||||||
// Drive the diplomaticclock graph of the DigitalTop (if present)
|
l.asyncClockGroupsNode
|
||||||
val simpleClockGroupSourceNode = chiptop.lazySystem match {
|
|
||||||
case l: BaseSubsystem if (p(SubsystemDriveAsyncClockGroupsKey).isEmpty) => {
|
|
||||||
val n = ClockGroupSourceNode(Seq(ClockGroupSourceParameters()))
|
|
||||||
l.asyncClockGroupsNode := n
|
|
||||||
Some(n)
|
|
||||||
}
|
|
||||||
case _ => None
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val aggregator = LazyModule(new ClockGroupAggregator("allClocks")).node
|
||||||
|
(chiptop.implicitClockSinkNode := ClockGroup() := aggregator)
|
||||||
|
(systemAsyncClockGroup := ClockGroupNamePrefixer() := aggregator)
|
||||||
|
|
||||||
|
val inputClockSource = ClockGroupSourceNode(Seq(ClockGroupSourceParameters()))
|
||||||
|
|
||||||
|
(aggregator
|
||||||
|
:= ClockGroupFrequencySpecifier(p(ClockFrequencyAssignersKey), p(DefaultClockFrequencyKey))
|
||||||
|
:= inputClockSource)
|
||||||
|
|
||||||
|
|
||||||
InModuleBody {
|
InModuleBody {
|
||||||
val clock = IO(Input(Clock())).suggestName("clock")
|
val (clockGroupBundle, clockGroupEdge) = inputClockSource.out.head
|
||||||
|
val input_clocks = IO(Input(RecordMap((clockGroupEdge.sink.members.map { m => (m.name.get, Clock()) }):_* )))
|
||||||
|
.suggestName("clocks")
|
||||||
val reset = IO(Input(Reset())).suggestName("reset")
|
val reset = IO(Input(Reset())).suggestName("reset")
|
||||||
|
|
||||||
implicitClockSourceNode.out.unzip._1.map { o =>
|
(clockGroupBundle.member.data zip input_clocks.data).foreach { case (clockBundle, inputClock) =>
|
||||||
o.clock := clock
|
clockBundle.clock := inputClock
|
||||||
o.reset := reset
|
|
||||||
}
|
}
|
||||||
|
|
||||||
simpleClockGroupSourceNode.map { n => n.out.unzip._1.map { out: ClockGroupBundle =>
|
// Assign resets. The synchronization scheme is still WIP.
|
||||||
out.member.data.foreach { o =>
|
for ((name, clockBundle) <- clockGroupBundle.member.elements) {
|
||||||
o.clock := clock
|
if (name.contains("core")) {
|
||||||
o.reset := reset
|
clockBundle.reset := ResetCatchAndSync(clockBundle.clock, reset.asBool)
|
||||||
|
} else {
|
||||||
|
clockBundle.reset := reset
|
||||||
}
|
}
|
||||||
}}
|
}
|
||||||
|
|
||||||
|
val pllConfig = new SimplePllConfiguration(clockGroupEdge.sink.members)
|
||||||
|
pllConfig.prettyPrint("FireSim RationalClockBridge")
|
||||||
|
val rationalClockSpecs = for ((sinkP, division) <- pllConfig.sinkDividerMap) yield {
|
||||||
|
RationalClock(sinkP.name.get, 1, division)
|
||||||
|
}
|
||||||
|
|
||||||
chiptop.harnessFunctions += ((th: HasHarnessSignalReferences) => {
|
chiptop.harnessFunctions += ((th: HasHarnessSignalReferences) => {
|
||||||
clock := th.harnessClock
|
|
||||||
reset := th.harnessReset
|
reset := th.harnessReset
|
||||||
Nil
|
input_clocks := p(ClockBridgeInstantiatorKey)
|
||||||
})
|
.getClockRecordOrInstantiate(rationalClockSpecs.toSeq, p(FireSimBaseClockNameKey))
|
||||||
}
|
Nil })
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
class WithFireSimRationalTileDomain(multiplier: Int, divisor: Int) extends Config((site, here, up) => {
|
|
||||||
case FireSimClockKey => FireSimClockParameters(Seq(RationalClock("TileDomain", multiplier, divisor)))
|
|
||||||
case ClockingSchemeKey => { chiptop: ChipTop =>
|
|
||||||
implicit val p = chiptop.p
|
|
||||||
|
|
||||||
val implicitClockSourceNode = ClockSourceNode(Seq(ClockSourceParameters()))
|
|
||||||
chiptop.implicitClockSinkNode := implicitClockSourceNode
|
|
||||||
|
|
||||||
// Drive the diplomaticclock graph of the DigitalTop (if present)
|
|
||||||
val simpleClockGroupSourceNode = chiptop.lazySystem match {
|
|
||||||
case l: BaseSubsystem if (p(SubsystemDriveAsyncClockGroupsKey).isEmpty) => {
|
|
||||||
val n = ClockGroupSourceNode(Seq(ClockGroupSourceParameters()))
|
|
||||||
l.asyncClockGroupsNode := n
|
|
||||||
Some(n)
|
|
||||||
}
|
|
||||||
case _ => None
|
|
||||||
}
|
|
||||||
|
|
||||||
InModuleBody {
|
|
||||||
val uncore_clock = IO(Input(Clock())).suggestName("uncore_clock")
|
|
||||||
val tile_clock = IO(Input(Clock())).suggestName("tile_clock")
|
|
||||||
val reset = IO(Input(Reset())).suggestName("reset")
|
|
||||||
|
|
||||||
implicitClockSourceNode.out.unzip._1.map { o =>
|
|
||||||
o.clock := uncore_clock
|
|
||||||
o.reset := reset
|
|
||||||
}
|
|
||||||
|
|
||||||
simpleClockGroupSourceNode.map { n => n.out.unzip._1.map { out: ClockGroupBundle =>
|
|
||||||
out.member.elements.map { case (name, data) =>
|
|
||||||
// This is mega hacks, how are you actually supposed to do this?
|
|
||||||
if (name.contains("core")) {
|
|
||||||
data.clock := tile_clock
|
|
||||||
data.reset := ResetCatchAndSync(tile_clock, reset.asBool)
|
|
||||||
} else {
|
|
||||||
data.clock := uncore_clock
|
|
||||||
data.reset := reset
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
|
|
||||||
chiptop.harnessFunctions += ((th: HasHarnessSignalReferences) => {
|
|
||||||
uncore_clock := th.harnessClock
|
|
||||||
reset := th.harnessReset
|
|
||||||
th match {
|
|
||||||
case f: FireSim => tile_clock := f.additionalClocks(0)
|
|
||||||
case _ => throw new Exception("FireSimMultiClock must be used with FireSim")
|
|
||||||
}
|
|
||||||
Nil
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
class FireSim(implicit val p: Parameters) extends RawModule with HasHarnessSignalReferences {
|
class FireSim(implicit val p: Parameters) extends RawModule with HasHarnessSignalReferences {
|
||||||
freechips.rocketchip.util.property.cover.setPropLib(new midas.passes.FireSimPropertyLibrary())
|
freechips.rocketchip.util.property.cover.setPropLib(new midas.passes.FireSimPropertyLibrary())
|
||||||
val clockBridge = Module(new RationalClockBridge(p(FireSimClockKey).additionalClocks:_*))
|
val harnessClock = Wire(Clock())
|
||||||
val harnessClock = clockBridge.io.clocks.head // This is the reference clock
|
|
||||||
val additionalClocks = clockBridge.io.clocks.tail
|
|
||||||
val harnessReset = WireInit(false.B)
|
val harnessReset = WireInit(false.B)
|
||||||
val peekPokeBridge = PeekPokeBridge(harnessClock, harnessReset)
|
val peekPokeBridge = PeekPokeBridge(harnessClock, harnessReset)
|
||||||
def dutReset = { require(false, "dutReset should not be used in Firesim"); false.B }
|
def dutReset = { require(false, "dutReset should not be used in Firesim"); false.B }
|
||||||
@@ -165,8 +163,7 @@ class FireSim(implicit val p: Parameters) extends RawModule with HasHarnessSigna
|
|||||||
d.harnessFunctions.foreach(_(this))
|
d.harnessFunctions.foreach(_(this))
|
||||||
ApplyHarnessBinders(this, d.lazySystem, p(HarnessBinders), d.portMap.toMap)
|
ApplyHarnessBinders(this, d.lazySystem, p(HarnessBinders), d.portMap.toMap)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
NodeIdx.increment()
|
NodeIdx.increment()
|
||||||
}
|
}
|
||||||
|
harnessClock := p(ClockBridgeInstantiatorKey).getClockRecord("implicit_clock").get
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -194,7 +194,7 @@ class FireSimArianeConfig extends Config(
|
|||||||
//* Multiclock Configurations
|
//* Multiclock Configurations
|
||||||
//*********************************************************************************/
|
//*********************************************************************************/
|
||||||
class FireSimMulticlockRocketConfig extends Config(
|
class FireSimMulticlockRocketConfig extends Config(
|
||||||
new WithFireSimRationalTileDomain(2, 1) ++
|
new chipyard.config.WithTileFrequency(6400.0) ++ //lol
|
||||||
new WithDefaultFireSimBridges ++
|
new WithDefaultFireSimBridges ++
|
||||||
new WithDefaultMemModel ++
|
new WithDefaultMemModel ++
|
||||||
new WithFireSimConfigTweaks ++
|
new WithFireSimConfigTweaks ++
|
||||||
|
|||||||
Submodule sims/firesim updated: c1cd3e5e70...4342b33301
Reference in New Issue
Block a user