mirror of
https://github.com/SpinalHDL/VexRiscv.git
synced 2025-01-03 03:43:39 -05:00
Merge remote-tracking branch 'origin/dev'
This commit is contained in:
commit
9a61ff8347
16 changed files with 415 additions and 254 deletions
|
@ -156,8 +156,8 @@ sudo make install
|
||||||
|
|
||||||
## CPU generation
|
## CPU generation
|
||||||
You can find two example CPU instances in:
|
You can find two example CPU instances in:
|
||||||
- src/main/scala/vexriscv/GenFull.scala
|
- src/main/scala/vexriscv/demo/GenFull.scala
|
||||||
- src/main/scala/vexriscv/GenSmallest.scala
|
- src/main/scala/vexriscv/demo/GenSmallest.scala
|
||||||
|
|
||||||
To generate the corresponding RTL as a VexRiscv.v file, run the following commands in the root directory of this repository:
|
To generate the corresponding RTL as a VexRiscv.v file, run the following commands in the root directory of this repository:
|
||||||
|
|
||||||
|
|
32
build.sbt
32
build.sbt
|
@ -1,25 +1,3 @@
|
||||||
//name := "VexRiscv"
|
|
||||||
//
|
|
||||||
//organization := "com.github.spinalhdl"
|
|
||||||
//
|
|
||||||
//version := "1.0.0"
|
|
||||||
//
|
|
||||||
//scalaVersion := "2.11.6"
|
|
||||||
//
|
|
||||||
//EclipseKeys.withSource := true
|
|
||||||
//
|
|
||||||
//libraryDependencies ++= Seq(
|
|
||||||
// "com.github.spinalhdl" % "spinalhdl-core_2.11" % "1.2.1",
|
|
||||||
// "com.github.spinalhdl" % "spinalhdl-lib_2.11" % "1.2.1",
|
|
||||||
// "org.scalatest" % "scalatest_2.11" % "2.2.1",
|
|
||||||
// "org.yaml" % "snakeyaml" % "1.8"
|
|
||||||
//)
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//addCompilerPlugin("org.scala-lang.plugins" % "scala-continuations-plugin_2.11.6" % "1.0.2")
|
|
||||||
//scalacOptions += "-P:continuations:enable"
|
|
||||||
//fork := true
|
|
||||||
|
|
||||||
lazy val root = (project in file(".")).
|
lazy val root = (project in file(".")).
|
||||||
settings(
|
settings(
|
||||||
|
@ -29,16 +7,16 @@ lazy val root = (project in file(".")).
|
||||||
version := "1.0.0"
|
version := "1.0.0"
|
||||||
)),
|
)),
|
||||||
libraryDependencies ++= Seq(
|
libraryDependencies ++= Seq(
|
||||||
"com.github.spinalhdl" % "spinalhdl-core_2.11" % "1.3.1",
|
"com.github.spinalhdl" % "spinalhdl-core_2.11" % "1.3.2",
|
||||||
"com.github.spinalhdl" % "spinalhdl-lib_2.11" % "1.3.1",
|
"com.github.spinalhdl" % "spinalhdl-lib_2.11" % "1.3.2",
|
||||||
"org.scalatest" % "scalatest_2.11" % "2.2.1",
|
"org.scalatest" % "scalatest_2.11" % "2.2.1",
|
||||||
"org.yaml" % "snakeyaml" % "1.8"
|
"org.yaml" % "snakeyaml" % "1.8"
|
||||||
),
|
),
|
||||||
name := "VexRiscv"
|
name := "VexRiscv"
|
||||||
)//.dependsOn(spinalHdlSim,spinalHdlCore,spinalHdlLib)
|
)//.dependsOn(spinalHdlSim,spinalHdlCore,spinalHdlLib)
|
||||||
//lazy val spinalHdlSim = ProjectRef(file("../SpinalHDL"), "SpinalHDL-sim")
|
//lazy val spinalHdlSim = ProjectRef(file("../SpinalHDL"), "sim")
|
||||||
//lazy val spinalHdlCore = ProjectRef(file("../SpinalHDL"), "SpinalHDL-core")
|
//lazy val spinalHdlCore = ProjectRef(file("../SpinalHDL"), "core")
|
||||||
//lazy val spinalHdlLib = ProjectRef(file("../SpinalHDL"), "SpinalHDL-lib")
|
//lazy val spinalHdlLib = ProjectRef(file("../SpinalHDL"), "lib")
|
||||||
|
|
||||||
|
|
||||||
fork := true
|
fork := true
|
|
@ -30,6 +30,13 @@ trait Pipeline {
|
||||||
filtered.length != 0
|
filtered.length != 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def serviceElse[T](clazz : Class[T], default : => T) : T = {
|
||||||
|
if(!serviceExist(clazz)) return default
|
||||||
|
val filtered = plugins.filter(o => clazz.isAssignableFrom(o.getClass))
|
||||||
|
assert(filtered.length == 1)
|
||||||
|
filtered.head.asInstanceOf[T]
|
||||||
|
}
|
||||||
|
|
||||||
def update[T](that : PipelineConfig[T], value : T) : Unit = configs(that) = value
|
def update[T](that : PipelineConfig[T], value : T) : Unit = configs(that) = value
|
||||||
def apply[T](that : PipelineConfig[T]) : T = configs(that).asInstanceOf[T]
|
def apply[T](that : PipelineConfig[T]) : T = configs(that).asInstanceOf[T]
|
||||||
|
|
||||||
|
|
|
@ -39,6 +39,10 @@ trait PrivilegeService{
|
||||||
def isUser(stage : Stage) : Bool
|
def isUser(stage : Stage) : Bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case class PrivilegeServiceDefault() extends PrivilegeService{
|
||||||
|
override def isUser(stage: Stage): Bool = False
|
||||||
|
}
|
||||||
|
|
||||||
trait InterruptionInhibitor{
|
trait InterruptionInhibitor{
|
||||||
def inhibateInterrupts() : Unit
|
def inhibateInterrupts() : Unit
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,8 +28,8 @@ import spinal.lib.eda.altera.{InterruptReceiverTag, ResetEmitterTag}
|
||||||
|
|
||||||
object TestsWorkspace {
|
object TestsWorkspace {
|
||||||
def main(args: Array[String]) {
|
def main(args: Array[String]) {
|
||||||
SpinalConfig(mergeAsyncProcess = false, anonymSignalPrefix = "zz_").generateVerilog {
|
def configFull = {
|
||||||
val configFull = VexRiscvConfig(
|
val config = VexRiscvConfig(
|
||||||
plugins = List(
|
plugins = List(
|
||||||
// new IBusSimplePlugin(
|
// new IBusSimplePlugin(
|
||||||
// resetVector = 0x80000000l,
|
// resetVector = 0x80000000l,
|
||||||
|
@ -65,6 +65,7 @@ object TestsWorkspace {
|
||||||
portTlbSize = 4
|
portTlbSize = 4
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
|
// ).newTightlyCoupledPort(TightlyCoupledPortParameter("iBusTc", a => a(30 downto 28) === 0x0 && a(5))),
|
||||||
// new DBusSimplePlugin(
|
// new DBusSimplePlugin(
|
||||||
// catchAddressMisaligned = true,
|
// catchAddressMisaligned = true,
|
||||||
// catchAccessFault = false,
|
// catchAccessFault = false,
|
||||||
|
@ -98,7 +99,7 @@ object TestsWorkspace {
|
||||||
ioRange = _(31 downto 28) === 0xF
|
ioRange = _(31 downto 28) === 0xF
|
||||||
),
|
),
|
||||||
new DecoderSimplePlugin(
|
new DecoderSimplePlugin(
|
||||||
catchIllegalInstruction = false
|
catchIllegalInstruction = true
|
||||||
),
|
),
|
||||||
new RegFilePlugin(
|
new RegFilePlugin(
|
||||||
regFileReadyKind = plugin.ASYNC,
|
regFileReadyKind = plugin.ASYNC,
|
||||||
|
@ -162,8 +163,28 @@ object TestsWorkspace {
|
||||||
new YamlPlugin("cpu0.yaml")
|
new YamlPlugin("cpu0.yaml")
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
config
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// import spinal.core.sim._
|
||||||
|
// SimConfig.withConfig(SpinalConfig(mergeAsyncProcess = false, anonymSignalPrefix = "zz_")).allOptimisation.compile(new VexRiscv(configFull)).doSimUntilVoid{ dut =>
|
||||||
|
// dut.clockDomain.forkStimulus(10)
|
||||||
|
// dut.clockDomain.forkSimSpeedPrinter(4)
|
||||||
|
// var iBus : InstructionCacheMemBus = null
|
||||||
|
//
|
||||||
|
// dut.plugins.foreach{
|
||||||
|
// case plugin: IBusCachedPlugin => iBus = plugin.iBus
|
||||||
|
// case _ =>
|
||||||
|
// }
|
||||||
|
// dut.clockDomain.onSamplings{
|
||||||
|
//// iBus.cmd.ready.randomize()
|
||||||
|
// iBus.rsp.data #= 0x13
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
SpinalConfig(mergeAsyncProcess = false, anonymSignalPrefix = "zz_").generateVerilog {
|
||||||
|
|
||||||
|
|
||||||
val toplevel = new VexRiscv(configFull)
|
val toplevel = new VexRiscv(configFull)
|
||||||
// val toplevel = new VexRiscv(configLight)
|
// val toplevel = new VexRiscv(configLight)
|
||||||
|
|
|
@ -258,10 +258,12 @@ class Briey(config: BrieyConfig) extends Component{
|
||||||
)
|
)
|
||||||
|
|
||||||
val gpioACtrl = Apb3Gpio(
|
val gpioACtrl = Apb3Gpio(
|
||||||
gpioWidth = 32
|
gpioWidth = 32,
|
||||||
|
withReadSync = true
|
||||||
)
|
)
|
||||||
val gpioBCtrl = Apb3Gpio(
|
val gpioBCtrl = Apb3Gpio(
|
||||||
gpioWidth = 32
|
gpioWidth = 32,
|
||||||
|
withReadSync = true
|
||||||
)
|
)
|
||||||
val timerCtrl = PinsecTimerCtrl()
|
val timerCtrl = PinsecTimerCtrl()
|
||||||
|
|
||||||
|
|
|
@ -276,7 +276,7 @@ case class Murax(config : MuraxConfig) extends Component{
|
||||||
|
|
||||||
//******** APB peripherals *********
|
//******** APB peripherals *********
|
||||||
val apbMapping = ArrayBuffer[(Apb3, SizeMapping)]()
|
val apbMapping = ArrayBuffer[(Apb3, SizeMapping)]()
|
||||||
val gpioACtrl = Apb3Gpio(gpioWidth = gpioWidth)
|
val gpioACtrl = Apb3Gpio(gpioWidth = gpioWidth, withReadSync = true)
|
||||||
io.gpioA <> gpioACtrl.io.gpio
|
io.gpioA <> gpioACtrl.io.gpio
|
||||||
apbMapping += gpioACtrl.io.apb -> (0x00000, 4 kB)
|
apbMapping += gpioACtrl.io.apb -> (0x00000, 4 kB)
|
||||||
|
|
||||||
|
|
|
@ -90,11 +90,13 @@ trait InstructionCacheCommons{
|
||||||
}
|
}
|
||||||
|
|
||||||
case class InstructionCacheCpuFetch(p : InstructionCacheConfig) extends Bundle with IMasterSlave with InstructionCacheCommons {
|
case class InstructionCacheCpuFetch(p : InstructionCacheConfig) extends Bundle with IMasterSlave with InstructionCacheCommons {
|
||||||
val isValid = Bool
|
val isValid = Bool()
|
||||||
val isStuck = Bool
|
val isStuck = Bool()
|
||||||
val isRemoved = Bool
|
val isRemoved = Bool()
|
||||||
val pc = UInt(p.addressWidth bits)
|
val pc = UInt(p.addressWidth bits)
|
||||||
val data = Bits(p.cpuDataWidth bits)
|
val data = Bits(p.cpuDataWidth bits)
|
||||||
|
val dataBypassValid = Bool()
|
||||||
|
val dataBypass = Bits(p.cpuDataWidth bits)
|
||||||
val mmuBus = MemoryTranslatorBus()
|
val mmuBus = MemoryTranslatorBus()
|
||||||
val physicalAddress = UInt(p.addressWidth bits)
|
val physicalAddress = UInt(p.addressWidth bits)
|
||||||
val cacheMiss, error, mmuMiss, illegalAccess,isUser = ifGen(!p.twoCycleCache)(Bool)
|
val cacheMiss, error, mmuMiss, illegalAccess,isUser = ifGen(!p.twoCycleCache)(Bool)
|
||||||
|
@ -102,7 +104,7 @@ case class InstructionCacheCpuFetch(p : InstructionCacheConfig) extends Bundle w
|
||||||
override def asMaster(): Unit = {
|
override def asMaster(): Unit = {
|
||||||
out(isValid, isStuck, isRemoved, pc)
|
out(isValid, isStuck, isRemoved, pc)
|
||||||
inWithNull(error,mmuMiss,illegalAccess,data, cacheMiss,physicalAddress)
|
inWithNull(error,mmuMiss,illegalAccess,data, cacheMiss,physicalAddress)
|
||||||
outWithNull(isUser)
|
outWithNull(isUser, dataBypass, dataBypassValid)
|
||||||
slaveWithNull(mmuBus)
|
slaveWithNull(mmuBus)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -381,21 +383,21 @@ class InstructionCache(p : InstructionCacheConfig) extends Component{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
val hit = if(!twoCycleRam) new Area{
|
val hit = (!twoCycleRam) generate new Area{
|
||||||
val hits = read.waysValues.map(way => way.tag.valid && way.tag.address === io.cpu.fetch.mmuBus.rsp.physicalAddress(tagRange))
|
val hits = read.waysValues.map(way => way.tag.valid && way.tag.address === io.cpu.fetch.mmuBus.rsp.physicalAddress(tagRange))
|
||||||
val valid = Cat(hits).orR
|
val valid = Cat(hits).orR
|
||||||
val id = OHToUInt(hits)
|
val id = OHToUInt(hits)
|
||||||
val error = read.waysValues.map(_.tag.error).read(id)
|
val error = read.waysValues.map(_.tag.error).read(id)
|
||||||
val data = read.waysValues.map(_.data).read(id)
|
val data = read.waysValues.map(_.data).read(id)
|
||||||
val word = data.subdivideIn(cpuDataWidth bits).read(io.cpu.fetch.pc(memWordToCpuWordRange))
|
val word = data.subdivideIn(cpuDataWidth bits).read(io.cpu.fetch.pc(memWordToCpuWordRange))
|
||||||
io.cpu.fetch.data := word
|
io.cpu.fetch.data := (io.cpu.fetch.dataBypassValid ? io.cpu.fetch.dataBypass | word)
|
||||||
if(twoCycleCache){
|
if(twoCycleCache){
|
||||||
io.cpu.decode.data := RegNextWhen(io.cpu.fetch.data,!io.cpu.decode.isStuck)
|
io.cpu.decode.data := RegNextWhen(io.cpu.fetch.data,!io.cpu.decode.isStuck)
|
||||||
}
|
}
|
||||||
} else null
|
}
|
||||||
|
|
||||||
if(twoCycleRam && wayCount == 1){
|
if(twoCycleRam && wayCount == 1){
|
||||||
io.cpu.fetch.data := read.waysValues.head.data.subdivideIn(cpuDataWidth bits).read(io.cpu.fetch.pc(memWordToCpuWordRange))
|
io.cpu.fetch.data := (io.cpu.fetch.dataBypassValid ? io.cpu.fetch.dataBypass | read.waysValues.head.data.subdivideIn(cpuDataWidth bits).read(io.cpu.fetch.pc(memWordToCpuWordRange)))
|
||||||
}
|
}
|
||||||
|
|
||||||
io.cpu.fetch.mmuBus.cmd.isValid := io.cpu.fetch.isValid
|
io.cpu.fetch.mmuBus.cmd.isValid := io.cpu.fetch.isValid
|
||||||
|
@ -405,7 +407,6 @@ class InstructionCache(p : InstructionCacheConfig) extends Component{
|
||||||
io.cpu.fetch.physicalAddress := io.cpu.fetch.mmuBus.rsp.physicalAddress
|
io.cpu.fetch.physicalAddress := io.cpu.fetch.mmuBus.rsp.physicalAddress
|
||||||
|
|
||||||
val resolution = ifGen(!twoCycleCache)( new Area{
|
val resolution = ifGen(!twoCycleCache)( new Area{
|
||||||
// def stage[T <: Data](that : T) = RegNextWhen(that,!io.cpu.decode.isStuck)
|
|
||||||
val mmuRsp = io.cpu.fetch.mmuBus.rsp
|
val mmuRsp = io.cpu.fetch.mmuBus.rsp
|
||||||
|
|
||||||
io.cpu.fetch.cacheMiss := !hit.valid
|
io.cpu.fetch.cacheMiss := !hit.valid
|
||||||
|
@ -432,17 +433,13 @@ class InstructionCache(p : InstructionCacheConfig) extends Component{
|
||||||
val error = tags(id).error
|
val error = tags(id).error
|
||||||
val data = fetchStage.read.waysValues.map(way => stage(way.data)).read(id)
|
val data = fetchStage.read.waysValues.map(way => stage(way.data)).read(id)
|
||||||
val word = data.subdivideIn(cpuDataWidth bits).read(io.cpu.decode.pc(memWordToCpuWordRange))
|
val word = data.subdivideIn(cpuDataWidth bits).read(io.cpu.decode.pc(memWordToCpuWordRange))
|
||||||
|
when(stage(io.cpu.fetch.dataBypassValid)){
|
||||||
|
word := stage(io.cpu.fetch.dataBypass)
|
||||||
|
}
|
||||||
io.cpu.decode.data := word
|
io.cpu.decode.data := word
|
||||||
}
|
}
|
||||||
|
|
||||||
io.cpu.decode.cacheMiss := !hit.valid
|
io.cpu.decode.cacheMiss := !hit.valid
|
||||||
// when( io.cpu.decode.isValid && io.cpu.decode.cacheMiss){
|
|
||||||
// io.cpu.prefetch.haltIt := True
|
|
||||||
// lineLoader.valid := True
|
|
||||||
// lineLoader.address := mmuRsp.physicalAddress //Could be optimise if mmu not used
|
|
||||||
// }
|
|
||||||
// when(io.cpu)
|
|
||||||
|
|
||||||
io.cpu.decode.error := hit.error
|
io.cpu.decode.error := hit.error
|
||||||
io.cpu.decode.mmuMiss := mmuRsp.miss
|
io.cpu.decode.mmuMiss := mmuRsp.miss
|
||||||
io.cpu.decode.illegalAccess := !mmuRsp.allowExecute || (io.cpu.decode.isUser && !mmuRsp.allowUser)
|
io.cpu.decode.illegalAccess := !mmuRsp.allowExecute || (io.cpu.decode.isUser && !mmuRsp.allowUser)
|
||||||
|
|
|
@ -271,6 +271,20 @@ class CsrPlugin(config: CsrPluginConfig) extends Plugin[VexRiscv] with Exception
|
||||||
val csrMapping = new CsrMapping()
|
val csrMapping = new CsrMapping()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
case class InterruptSource(var cond : Bool, id : Int)
|
||||||
|
case class InterruptPrivilege(privilege : Int){
|
||||||
|
var privilegeCond : Bool = null
|
||||||
|
val sources = ArrayBuffer[InterruptSource]()
|
||||||
|
}
|
||||||
|
|
||||||
|
def getInterruptPrivilege(privilege : Int) = customInterrupts.getOrElseUpdate(privilege, InterruptPrivilege(privilege))
|
||||||
|
var customInterrupts = mutable.LinkedHashMap[Int, InterruptPrivilege]()
|
||||||
|
def addInterrupt(cond : Bool, id : Int, privilege : Int): Unit = {
|
||||||
|
getInterruptPrivilege(privilege).sources += InterruptSource(cond, id)
|
||||||
|
}
|
||||||
|
def createInterrupt(id : Int, privilege : Int) : Bool = { val ret = Bool(); addInterrupt(ret, id, privilege); ret}
|
||||||
|
|
||||||
override def r(csrAddress: Int, bitOffset: Int, that: Data): Unit = csrMapping.r(csrAddress, bitOffset, that)
|
override def r(csrAddress: Int, bitOffset: Int, that: Data): Unit = csrMapping.r(csrAddress, bitOffset, that)
|
||||||
override def w(csrAddress: Int, bitOffset: Int, that: Data): Unit = csrMapping.w(csrAddress, bitOffset, that)
|
override def w(csrAddress: Int, bitOffset: Int, that: Data): Unit = csrMapping.w(csrAddress, bitOffset, that)
|
||||||
override def onWrite(csrAddress: Int)(body: => Unit): Unit = csrMapping.onWrite(csrAddress)(body)
|
override def onWrite(csrAddress: Int)(body: => Unit): Unit = csrMapping.onWrite(csrAddress)(body)
|
||||||
|
@ -338,6 +352,11 @@ class CsrPlugin(config: CsrPluginConfig) extends Plugin[VexRiscv] with Exception
|
||||||
|
|
||||||
allowInterrupts = True
|
allowInterrupts = True
|
||||||
allowException = True
|
allowException = True
|
||||||
|
|
||||||
|
for (privilege <- customInterrupts.values;
|
||||||
|
source <- privilege.sources){
|
||||||
|
source.cond = source.cond.pull()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def inhibateInterrupts() : Unit = allowInterrupts := False
|
def inhibateInterrupts() : Unit = allowInterrupts := False
|
||||||
|
@ -500,20 +519,22 @@ class CsrPlugin(config: CsrPluginConfig) extends Plugin[VexRiscv] with Exception
|
||||||
minstret := minstret + 1
|
minstret := minstret + 1
|
||||||
}
|
}
|
||||||
|
|
||||||
case class InterruptSource(cond : Bool, id : Int)
|
|
||||||
case class InterruptPrivilege(privilege : Int, privilegeCond : Bool, sources : ArrayBuffer[InterruptSource])
|
if(supervisorGen) {
|
||||||
val interruptModel = ArrayBuffer[InterruptPrivilege]()
|
getInterruptPrivilege(1).privilegeCond = sstatus.SIE && privilege <= "01"
|
||||||
if(supervisorGen) interruptModel += InterruptPrivilege(1, sstatus.SIE && privilege <= "01", ArrayBuffer(
|
getInterruptPrivilege(1).sources ++= List(
|
||||||
InterruptSource(sip.STIP && sie.STIE, 5),
|
InterruptSource(sip.STIP && sie.STIE, 5),
|
||||||
InterruptSource(sip.SSIP && sie.SSIE, 1),
|
InterruptSource(sip.SSIP && sie.SSIE, 1),
|
||||||
InterruptSource(sip.SEIP && sie.SEIE, 9)
|
InterruptSource(sip.SEIP && sie.SEIE, 9)
|
||||||
))
|
)
|
||||||
|
}
|
||||||
|
|
||||||
interruptModel += InterruptPrivilege(3, mstatus.MIE , ArrayBuffer(
|
getInterruptPrivilege(3).privilegeCond = mstatus.MIE
|
||||||
|
getInterruptPrivilege(3).sources ++= List(
|
||||||
InterruptSource(mip.MTIP && mie.MTIE, 7),
|
InterruptSource(mip.MTIP && mie.MTIE, 7),
|
||||||
InterruptSource(mip.MSIP && mie.MSIE, 3),
|
InterruptSource(mip.MSIP && mie.MSIE, 3),
|
||||||
InterruptSource(mip.MEIP && mie.MEIE, 11)
|
InterruptSource(mip.MEIP && mie.MEIE, 11)
|
||||||
))
|
)
|
||||||
|
|
||||||
case class DelegatorModel(value : Bits, source : Int, target : Int)
|
case class DelegatorModel(value : Bits, source : Int, target : Int)
|
||||||
// def solveDelegators(delegators : Seq[DelegatorModel], id : Int, lowerBound : Int): UInt = {
|
// def solveDelegators(delegators : Seq[DelegatorModel], id : Int, lowerBound : Int): UInt = {
|
||||||
|
@ -621,7 +642,7 @@ class CsrPlugin(config: CsrPluginConfig) extends Plugin[VexRiscv] with Exception
|
||||||
val interrupt = False
|
val interrupt = False
|
||||||
val interruptCode = UInt(4 bits).assignDontCare().addTag(Verilator.public)
|
val interruptCode = UInt(4 bits).assignDontCare().addTag(Verilator.public)
|
||||||
val interruptDelegatorHit = interruptDelegators.map(d => (d -> False)).toMap
|
val interruptDelegatorHit = interruptDelegators.map(d => (d -> False)).toMap
|
||||||
for(model <- interruptModel){
|
for(model <- customInterrupts.values.toSeq.sortBy(_.privilege)){
|
||||||
when(model.privilegeCond){
|
when(model.privilegeCond){
|
||||||
when(model.sources.map(_.cond).orR){
|
when(model.sources.map(_.cond).orR){
|
||||||
interrupt := True
|
interrupt := True
|
||||||
|
|
|
@ -84,6 +84,25 @@ class DecoderSimplePlugin(catchIllegalInstruction : Boolean = false, forceLegalI
|
||||||
|
|
||||||
val stageables = (encodings.flatMap(_._2.map(_._1)) ++ defaults.map(_._1)).toSet.toList
|
val stageables = (encodings.flatMap(_._2.map(_._1)) ++ defaults.map(_._1)).toSet.toList
|
||||||
|
|
||||||
|
val stupidDecoder = false
|
||||||
|
if(stupidDecoder){
|
||||||
|
if (catchIllegalInstruction || forceLegalInstructionComputation) insert(LEGAL_INSTRUCTION) := False
|
||||||
|
for(stageable <- stageables){
|
||||||
|
if(defaults.contains(stageable)){
|
||||||
|
insert(stageable).assignFrom(defaults(stageable))
|
||||||
|
} else {
|
||||||
|
insert(stageable).assignDontCare()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for((key, tasks) <- encodings){
|
||||||
|
when(input(INSTRUCTION) === key){
|
||||||
|
if (catchIllegalInstruction || forceLegalInstructionComputation) insert(LEGAL_INSTRUCTION) := True
|
||||||
|
for((stageable, value) <- tasks){
|
||||||
|
insert(stageable).assignFrom(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
var offset = 0
|
var offset = 0
|
||||||
var defaultValue, defaultCare = BigInt(0)
|
var defaultValue, defaultCare = BigInt(0)
|
||||||
val offsetOf = mutable.HashMap[Stageable[_ <: BaseType], Int]()
|
val offsetOf = mutable.HashMap[Stageable[_ <: BaseType], Int]()
|
||||||
|
@ -123,7 +142,6 @@ class DecoderSimplePlugin(catchIllegalInstruction : Boolean = false, forceLegalI
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// logic implementation
|
// logic implementation
|
||||||
val decodedBits = Bits(stageables.foldLeft(0)(_ + _.dataType.getBitsWidth) bits)
|
val decodedBits = Bits(stageables.foldLeft(0)(_ + _.dataType.getBitsWidth) bits)
|
||||||
decodedBits := Symplify(input(INSTRUCTION), spec, decodedBits.getWidth)
|
decodedBits := Symplify(input(INSTRUCTION), spec, decodedBits.getWidth)
|
||||||
|
@ -137,7 +155,7 @@ class DecoderSimplePlugin(catchIllegalInstruction : Boolean = false, forceLegalI
|
||||||
// insert(e).assignFromBits(RegNext(decodedBits(offset, e.dataType.getBitsWidth bits)))
|
// insert(e).assignFromBits(RegNext(decodedBits(offset, e.dataType.getBitsWidth bits)))
|
||||||
offset += e.dataType.getBitsWidth
|
offset += e.dataType.getBitsWidth
|
||||||
})
|
})
|
||||||
|
}
|
||||||
|
|
||||||
if(catchIllegalInstruction){
|
if(catchIllegalInstruction){
|
||||||
decodeExceptionPort.valid := arbitration.isValid && input(INSTRUCTION_READY) && !input(LEGAL_INSTRUCTION) // ?? HalitIt to alow decoder stage to wait valid data from 2 stages cache cache ??
|
decodeExceptionPort.valid := arbitration.isValid && input(INSTRUCTION_READY) && !input(LEGAL_INSTRUCTION) // ?? HalitIt to alow decoder stage to wait valid data from 2 stages cache cache ??
|
||||||
|
@ -149,8 +167,8 @@ class DecoderSimplePlugin(catchIllegalInstruction : Boolean = false, forceLegalI
|
||||||
def bench(toplevel : VexRiscv): Unit ={
|
def bench(toplevel : VexRiscv): Unit ={
|
||||||
toplevel.rework{
|
toplevel.rework{
|
||||||
import toplevel.config._
|
import toplevel.config._
|
||||||
toplevel.getAllIo.foreach{io =>
|
toplevel.getAllIo.toList.foreach{io =>
|
||||||
if(io.isInput) io.assignDontCare()
|
if(io.isInput) { io.assignDontCare()}
|
||||||
io.setAsDirectionLess()
|
io.setAsDirectionLess()
|
||||||
}
|
}
|
||||||
toplevel.decode.input(INSTRUCTION).removeAssignments()
|
toplevel.decode.input(INSTRUCTION).removeAssignments()
|
||||||
|
|
|
@ -5,10 +5,26 @@ import vexriscv.ip._
|
||||||
import spinal.core._
|
import spinal.core._
|
||||||
import spinal.lib._
|
import spinal.lib._
|
||||||
|
|
||||||
|
import scala.collection.mutable.ArrayBuffer
|
||||||
|
|
||||||
//class IBusCachedPlugin(config : InstructionCacheConfig, memoryTranslatorPortConfig : Any = null) extends Plugin[VexRiscv] {
|
//class IBusCachedPlugin(config : InstructionCacheConfig, memoryTranslatorPortConfig : Any = null) extends Plugin[VexRiscv] {
|
||||||
// var iBus : InstructionCacheMemBus = null
|
// var iBus : InstructionCacheMemBus = null
|
||||||
// override def build(pipeline: VexRiscv): Unit = ???
|
// override def build(pipeline: VexRiscv): Unit = ???
|
||||||
//}
|
//}
|
||||||
|
|
||||||
|
case class TightlyCoupledBus() extends Bundle with IMasterSlave {
|
||||||
|
val enable = Bool()
|
||||||
|
val address = UInt(32 bits)
|
||||||
|
val data = Bits(32 bits)
|
||||||
|
|
||||||
|
override def asMaster(): Unit = {
|
||||||
|
out(enable, address)
|
||||||
|
in(data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
case class TightlyCoupledPortParameter(name : String, hit : UInt => Bool)
|
||||||
|
case class TightlyCoupledPort(p : TightlyCoupledPortParameter, var bus : TightlyCoupledBus)
|
||||||
class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
relaxedPcCalculation : Boolean = false,
|
relaxedPcCalculation : Boolean = false,
|
||||||
prediction : BranchPrediction = NONE,
|
prediction : BranchPrediction = NONE,
|
||||||
|
@ -17,7 +33,8 @@ class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
keepPcPlus4 : Boolean = false,
|
keepPcPlus4 : Boolean = false,
|
||||||
config : InstructionCacheConfig,
|
config : InstructionCacheConfig,
|
||||||
memoryTranslatorPortConfig : Any = null,
|
memoryTranslatorPortConfig : Any = null,
|
||||||
injectorStage : Boolean = false) extends IBusFetcherImpl(
|
injectorStage : Boolean = false,
|
||||||
|
withoutInjectorStage : Boolean = false) extends IBusFetcherImpl(
|
||||||
resetVector = resetVector,
|
resetVector = resetVector,
|
||||||
keepPcPlus4 = keepPcPlus4,
|
keepPcPlus4 = keepPcPlus4,
|
||||||
decodePcGen = compressedGen,
|
decodePcGen = compressedGen,
|
||||||
|
@ -27,15 +44,25 @@ class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
injectorReadyCutGen = false,
|
injectorReadyCutGen = false,
|
||||||
prediction = prediction,
|
prediction = prediction,
|
||||||
historyRamSizeLog2 = historyRamSizeLog2,
|
historyRamSizeLog2 = historyRamSizeLog2,
|
||||||
injectorStage = !config.twoCycleCache || injectorStage){
|
injectorStage = (!config.twoCycleCache && !withoutInjectorStage) || injectorStage){
|
||||||
import config._
|
import config._
|
||||||
|
|
||||||
|
assert(!(withoutInjectorStage && injectorStage))
|
||||||
|
|
||||||
var iBus : InstructionCacheMemBus = null
|
var iBus : InstructionCacheMemBus = null
|
||||||
var mmuBus : MemoryTranslatorBus = null
|
var mmuBus : MemoryTranslatorBus = null
|
||||||
var privilegeService : PrivilegeService = null
|
var privilegeService : PrivilegeService = null
|
||||||
var redoBranch : Flow[UInt] = null
|
var redoBranch : Flow[UInt] = null
|
||||||
var decodeExceptionPort : Flow[ExceptionCause] = null
|
var decodeExceptionPort : Flow[ExceptionCause] = null
|
||||||
|
val tightlyCoupledPorts = ArrayBuffer[TightlyCoupledPort]()
|
||||||
|
|
||||||
|
|
||||||
|
def newTightlyCoupledPort(p : TightlyCoupledPortParameter) = {
|
||||||
|
val port = TightlyCoupledPort(p, null)
|
||||||
|
tightlyCoupledPorts += port
|
||||||
|
this
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
object FLUSH_ALL extends Stageable(Bool)
|
object FLUSH_ALL extends Stageable(Bool)
|
||||||
object IBUS_ACCESS_ERROR extends Stageable(Bool)
|
object IBUS_ACCESS_ERROR extends Stageable(Bool)
|
||||||
|
@ -66,8 +93,7 @@ class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
if(pipeline.serviceExist(classOf[MemoryTranslator]))
|
if(pipeline.serviceExist(classOf[MemoryTranslator]))
|
||||||
mmuBus = pipeline.service(classOf[MemoryTranslator]).newTranslationPort(MemoryTranslatorPort.PRIORITY_INSTRUCTION, memoryTranslatorPortConfig)
|
mmuBus = pipeline.service(classOf[MemoryTranslator]).newTranslationPort(MemoryTranslatorPort.PRIORITY_INSTRUCTION, memoryTranslatorPortConfig)
|
||||||
|
|
||||||
if(pipeline.serviceExist(classOf[PrivilegeService]))
|
privilegeService = pipeline.serviceElse(classOf[PrivilegeService], PrivilegeServiceDefault())
|
||||||
privilegeService = pipeline.service(classOf[PrivilegeService])
|
|
||||||
|
|
||||||
if(pipeline.serviceExist(classOf[ReportService])){
|
if(pipeline.serviceExist(classOf[ReportService])){
|
||||||
val report = pipeline.service(classOf[ReportService])
|
val report = pipeline.service(classOf[ReportService])
|
||||||
|
@ -98,54 +124,65 @@ class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
val cache = new InstructionCache(IBusCachedPlugin.this.config)
|
val cache = new InstructionCache(IBusCachedPlugin.this.config)
|
||||||
iBus = master(new InstructionCacheMemBus(IBusCachedPlugin.this.config)).setName("iBus")
|
iBus = master(new InstructionCacheMemBus(IBusCachedPlugin.this.config)).setName("iBus")
|
||||||
iBus <> cache.io.mem
|
iBus <> cache.io.mem
|
||||||
iBus.cmd.address.allowOverride := cache.io.mem.cmd.address // - debugAddressOffset
|
iBus.cmd.address.allowOverride := cache.io.mem.cmd.address
|
||||||
|
|
||||||
val stageOffset = if(relaxedPcCalculation) 1 else 0
|
val stageOffset = if(relaxedPcCalculation) 1 else 0
|
||||||
def stages = iBusRsp.stages.drop(stageOffset)
|
def stages = iBusRsp.stages.drop(stageOffset)
|
||||||
|
|
||||||
|
tightlyCoupledPorts.foreach(p => p.bus = master(TightlyCoupledBus()).setName(p.p.name))
|
||||||
|
|
||||||
|
val s0 = new Area {
|
||||||
|
//address decoding
|
||||||
|
val tightlyCoupledHits = Vec(tightlyCoupledPorts.map(_.p.hit(stages(0).input.payload)))
|
||||||
|
val tightlyCoupledHit = tightlyCoupledHits.orR
|
||||||
|
|
||||||
|
for((port, hit) <- (tightlyCoupledPorts, tightlyCoupledHits).zipped){
|
||||||
|
port.bus.enable := stages(0).input.fire && hit
|
||||||
|
port.bus.address := stages(0).input.payload(31 downto 2) @@ U"00"
|
||||||
|
}
|
||||||
|
|
||||||
//Connect prefetch cache side
|
//Connect prefetch cache side
|
||||||
cache.io.cpu.prefetch.isValid := stages(0).input.valid
|
cache.io.cpu.prefetch.isValid := stages(0).input.valid && !tightlyCoupledHit
|
||||||
cache.io.cpu.prefetch.pc := stages(0).input.payload
|
cache.io.cpu.prefetch.pc := stages(0).input.payload
|
||||||
stages(0).halt setWhen (cache.io.cpu.prefetch.haltIt)
|
stages(0).halt setWhen (cache.io.cpu.prefetch.haltIt)
|
||||||
|
|
||||||
|
|
||||||
cache.io.cpu.fetch.isRemoved := flush
|
cache.io.cpu.fetch.isRemoved := flush
|
||||||
val iBusRspOutputHalt = False
|
|
||||||
if (mmuBus != null) {
|
|
||||||
cache.io.cpu.fetch.mmuBus <> mmuBus
|
|
||||||
(if(twoCycleCache) stages(1).halt else iBusRspOutputHalt) setWhen(mmuBus.cmd.isValid && !mmuBus.rsp.hit && !mmuBus.rsp.miss)
|
|
||||||
} else {
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.physicalAddress := cache.io.cpu.fetch.mmuBus.cmd.virtualAddress
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.allowExecute := True
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.allowRead := True
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.allowWrite := True
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.allowUser := True
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.isIoAccess := False
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.miss := False
|
|
||||||
cache.io.cpu.fetch.mmuBus.rsp.hit := False
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
val s1 = new Area {
|
||||||
|
val tightlyCoupledHits = RegNextWhen(s0.tightlyCoupledHits, stages(1).input.ready)
|
||||||
|
val tightlyCoupledHit = RegNextWhen(s0.tightlyCoupledHit, stages(1).input.ready)
|
||||||
|
|
||||||
|
cache.io.cpu.fetch.dataBypassValid := tightlyCoupledHit
|
||||||
|
cache.io.cpu.fetch.dataBypass := (if(tightlyCoupledPorts.isEmpty) B(0) else MuxOH(tightlyCoupledHits, tightlyCoupledPorts.map(e => CombInit(e.bus.data))))
|
||||||
|
|
||||||
//Connect fetch cache side
|
//Connect fetch cache side
|
||||||
cache.io.cpu.fetch.isValid := stages(1).input.valid
|
cache.io.cpu.fetch.isValid := stages(1).input.valid && !tightlyCoupledHit
|
||||||
cache.io.cpu.fetch.isStuck := !stages(1).input.ready
|
cache.io.cpu.fetch.isStuck := !stages(1).input.ready
|
||||||
cache.io.cpu.fetch.pc := stages(1).input.payload
|
cache.io.cpu.fetch.pc := stages(1).input.payload
|
||||||
|
|
||||||
|
if (!twoCycleCache) {
|
||||||
|
cache.io.cpu.fetch.isUser := privilegeService.isUser(decode)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if(twoCycleCache){
|
val s2 = twoCycleCache generate new Area {
|
||||||
cache.io.cpu.decode.isValid := stages(2).input.valid
|
val tightlyCoupledHit = RegNextWhen(s1.tightlyCoupledHit, stages(2).input.ready)
|
||||||
|
cache.io.cpu.decode.isValid := stages(2).input.valid && !tightlyCoupledHit
|
||||||
cache.io.cpu.decode.isStuck := !stages(2).input.ready
|
cache.io.cpu.decode.isStuck := !stages(2).input.ready
|
||||||
cache.io.cpu.decode.pc := stages(2).input.payload
|
cache.io.cpu.decode.pc := stages(2).input.payload
|
||||||
cache.io.cpu.decode.isUser := (if (privilegeService != null) privilegeService.isUser(decode) else False)
|
cache.io.cpu.decode.isUser := privilegeService.isUser(decode)
|
||||||
|
|
||||||
if ((!twoCycleRam || wayCount == 1) && !compressedGen && !injectorStage) {
|
if ((!twoCycleRam || wayCount == 1) && !compressedGen && !injectorStage) {
|
||||||
decode.insert(INSTRUCTION_ANTICIPATED) := Mux(decode.arbitration.isStuck, decode.input(INSTRUCTION), cache.io.cpu.fetch.data)
|
decode.insert(INSTRUCTION_ANTICIPATED) := Mux(decode.arbitration.isStuck, decode.input(INSTRUCTION), cache.io.cpu.fetch.data)
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
cache.io.cpu.fetch.isUser := (if (privilegeService != null) privilegeService.isUser(decode) else False)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val rsp = new Area {
|
||||||
|
val iBusRspOutputHalt = False
|
||||||
|
|
||||||
|
|
||||||
// val missHalt = cache.io.cpu.fetch.isValid && cache.io.cpu.fetch.cacheMiss
|
|
||||||
val cacheRsp = if (twoCycleCache) cache.io.cpu.decode else cache.io.cpu.fetch
|
val cacheRsp = if (twoCycleCache) cache.io.cpu.decode else cache.io.cpu.fetch
|
||||||
val cacheRspArbitration = stages(if (twoCycleCache) 2 else 1)
|
val cacheRspArbitration = stages(if (twoCycleCache) 2 else 1)
|
||||||
var issueDetected = False
|
var issueDetected = False
|
||||||
|
@ -156,11 +193,12 @@ class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//Refill / redo
|
||||||
assert(decodePcGen == compressedGen)
|
assert(decodePcGen == compressedGen)
|
||||||
cache.io.cpu.fill.valid := redoFetch
|
cache.io.cpu.fill.valid := redoFetch
|
||||||
|
cache.io.cpu.fill.payload := cacheRsp.physicalAddress
|
||||||
redoBranch.valid := redoFetch
|
redoBranch.valid := redoFetch
|
||||||
redoBranch.payload := (if (decodePcGen) decode.input(PC) else cacheRsp.pc)
|
redoBranch.payload := (if (decodePcGen) decode.input(PC) else cacheRsp.pc)
|
||||||
cache.io.cpu.fill.payload := cacheRsp.physicalAddress
|
|
||||||
|
|
||||||
if (catchSomething) {
|
if (catchSomething) {
|
||||||
val accessFault = if (catchAccessFault) cacheRsp.error else False
|
val accessFault = if (catchAccessFault) cacheRsp.error else False
|
||||||
|
@ -177,10 +215,25 @@ class IBusCachedPlugin(resetVector : BigInt = 0x80000000l,
|
||||||
}
|
}
|
||||||
|
|
||||||
cacheRspArbitration.halt setWhen (issueDetected || iBusRspOutputHalt)
|
cacheRspArbitration.halt setWhen (issueDetected || iBusRspOutputHalt)
|
||||||
iBusRsp.output.arbitrationFrom(cacheRspArbitration.output)
|
iBusRsp.output.valid := cacheRspArbitration.output.valid
|
||||||
|
cacheRspArbitration.output.ready := iBusRsp.output.ready
|
||||||
iBusRsp.output.rsp.inst := cacheRsp.data
|
iBusRsp.output.rsp.inst := cacheRsp.data
|
||||||
iBusRsp.output.pc := cacheRspArbitration.output.payload
|
iBusRsp.output.pc := cacheRspArbitration.output.payload
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mmuBus != null) {
|
||||||
|
cache.io.cpu.fetch.mmuBus <> mmuBus
|
||||||
|
(if (twoCycleCache) stages(1).halt else rsp.iBusRspOutputHalt) setWhen (mmuBus.cmd.isValid && !mmuBus.rsp.hit && !mmuBus.rsp.miss)
|
||||||
|
} else {
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.physicalAddress := cache.io.cpu.fetch.mmuBus.cmd.virtualAddress
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.allowExecute := True
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.allowRead := True
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.allowWrite := True
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.allowUser := True
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.isIoAccess := False
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.miss := False
|
||||||
|
cache.io.cpu.fetch.mmuBus.rsp.hit := False
|
||||||
|
}
|
||||||
|
|
||||||
val flushStage = if(memory != null) memory else execute
|
val flushStage = if(memory != null) memory else execute
|
||||||
flushStage plug new Area {
|
flushStage plug new Area {
|
||||||
|
|
|
@ -1192,6 +1192,43 @@ public:
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef IBUS_TC
|
||||||
|
|
||||||
|
class IBusTc : public SimElement{
|
||||||
|
public:
|
||||||
|
|
||||||
|
uint32_t nextData;
|
||||||
|
|
||||||
|
Workspace *ws;
|
||||||
|
VVexRiscv* top;
|
||||||
|
IBusTc(Workspace* ws){
|
||||||
|
this->ws = ws;
|
||||||
|
this->top = ws->top;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void onReset(){
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void preCycle(){
|
||||||
|
if (top->iBusTc_enable) {
|
||||||
|
if((top->iBusTc_address & 0x70000000) != 0 || (top->iBusTc_address & 0x20) == 0){
|
||||||
|
printf("IBusTc access out of range\n");
|
||||||
|
ws->fail();
|
||||||
|
}
|
||||||
|
bool error_next;
|
||||||
|
ws->iBusAccess(top->iBusTc_address, &nextData,&error_next);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void postCycle(){
|
||||||
|
top->iBusTc_data = nextData;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
#ifdef IBUS_SIMPLE_AVALON
|
#ifdef IBUS_SIMPLE_AVALON
|
||||||
|
|
||||||
struct IBusSimpleAvalonRsp{
|
struct IBusSimpleAvalonRsp{
|
||||||
|
@ -1274,6 +1311,12 @@ public:
|
||||||
bool error;
|
bool error;
|
||||||
top->iBus_rsp_valid = 0;
|
top->iBus_rsp_valid = 0;
|
||||||
if(pendingCount != 0 && (!ws->iStall || VL_RANDOM_I(7) < 100)){
|
if(pendingCount != 0 && (!ws->iStall || VL_RANDOM_I(7) < 100)){
|
||||||
|
#ifdef IBUS_TC
|
||||||
|
if((address & 0x70000000) == 0 && (address & 0x20) != 0){
|
||||||
|
printf("IBUS_CACHED access out of range\n");
|
||||||
|
ws->fail();
|
||||||
|
}
|
||||||
|
#endif
|
||||||
ws->iBusAccess(address,&top->iBus_rsp_payload_data,&error);
|
ws->iBusAccess(address,&top->iBus_rsp_payload_data,&error);
|
||||||
top->iBus_rsp_payload_error = error;
|
top->iBus_rsp_payload_error = error;
|
||||||
pendingCount--;
|
pendingCount--;
|
||||||
|
@ -1960,6 +2003,11 @@ void Workspace::fillSimELements(){
|
||||||
#if defined(IBUS_CACHED_WISHBONE) || defined(IBUS_SIMPLE_WISHBONE)
|
#if defined(IBUS_CACHED_WISHBONE) || defined(IBUS_SIMPLE_WISHBONE)
|
||||||
simElements.push_back(new IBusCachedWishbone(this));
|
simElements.push_back(new IBusCachedWishbone(this));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef IBUS_TC
|
||||||
|
simElements.push_back(new IBusTc(this));
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef DBUS_SIMPLE
|
#ifdef DBUS_SIMPLE
|
||||||
simElements.push_back(new DBusSimple(this));
|
simElements.push_back(new DBusSimple(this));
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
DEBUG?=no
|
DEBUG?=no
|
||||||
|
|
||||||
IBUS?=CACHED
|
IBUS?=CACHED
|
||||||
|
IBUS_TC?=no
|
||||||
DBUS?=CACHED
|
DBUS?=CACHED
|
||||||
TRACE?=no
|
TRACE?=no
|
||||||
TRACE_ACCESS?=no
|
TRACE_ACCESS?=no
|
||||||
|
@ -42,7 +43,7 @@ ADDCFLAGS += -CFLAGS -DTHREAD_COUNT=${THREAD_COUNT}
|
||||||
ifeq ($(DEBUG),yes)
|
ifeq ($(DEBUG),yes)
|
||||||
ADDCFLAGS += -CFLAGS -O0 -CFLAGS -g
|
ADDCFLAGS += -CFLAGS -O0 -CFLAGS -g
|
||||||
else
|
else
|
||||||
ADDCFLAGS += -CFLAGS -O3
|
ADDCFLAGS += -CFLAGS -O3 -O3
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|
||||||
|
@ -58,6 +59,11 @@ ifneq ($(RUN_HEX),no)
|
||||||
ADDCFLAGS += -CFLAGS -DRUN_HEX='\"$(RUN_HEX)\"'
|
ADDCFLAGS += -CFLAGS -DRUN_HEX='\"$(RUN_HEX)\"'
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|
||||||
|
ifeq ($(IBUS_TC),yes)
|
||||||
|
ADDCFLAGS += -CFLAGS -DIBUS_TC=yes
|
||||||
|
endif
|
||||||
|
|
||||||
ifeq ($(COMPRESSED),yes)
|
ifeq ($(COMPRESSED),yes)
|
||||||
ADDCFLAGS += -CFLAGS -DCOMPRESSED
|
ADDCFLAGS += -CFLAGS -DCOMPRESSED
|
||||||
endif
|
endif
|
||||||
|
|
|
@ -284,6 +284,8 @@ class IBusDimension extends VexRiscvDimension("IBus") {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
val compressed = r.nextBoolean()
|
val compressed = r.nextBoolean()
|
||||||
|
val tighlyCoupled = r.nextBoolean()
|
||||||
|
// val tighlyCoupled = false
|
||||||
val prediction = random(r, List(NONE, STATIC, DYNAMIC, DYNAMIC_TARGET))
|
val prediction = random(r, List(NONE, STATIC, DYNAMIC, DYNAMIC_TARGET))
|
||||||
val catchAll = universes.contains(VexRiscvUniverse.CATCH_ALL)
|
val catchAll = universes.contains(VexRiscvUniverse.CATCH_ALL)
|
||||||
val relaxedPcCalculation, twoCycleCache, injectorStage = r.nextBoolean()
|
val relaxedPcCalculation, twoCycleCache, injectorStage = r.nextBoolean()
|
||||||
|
@ -295,9 +297,10 @@ class IBusDimension extends VexRiscvDimension("IBus") {
|
||||||
wayCount = 1 << r.nextInt(3)
|
wayCount = 1 << r.nextInt(3)
|
||||||
}while(cacheSize/wayCount < 512)
|
}while(cacheSize/wayCount < 512)
|
||||||
|
|
||||||
new VexRiscvPosition("Cached" + (if(twoCycleCache) "2cc" else "") + (if(injectorStage) "Injstage" else "") + (if(twoCycleRam) "2cr" else "") + "S" + cacheSize + "W" + wayCount + (if(relaxedPcCalculation) "Relax" else "") + (if(compressed) "Rvc" else "") + prediction.getClass.getTypeName().replace("$","")) with InstructionAnticipatedPosition{
|
new VexRiscvPosition("Cached" + (if(twoCycleCache) "2cc" else "") + (if(injectorStage) "Injstage" else "") + (if(twoCycleRam) "2cr" else "") + "S" + cacheSize + "W" + wayCount + (if(relaxedPcCalculation) "Relax" else "") + (if(compressed) "Rvc" else "") + prediction.getClass.getTypeName().replace("$","")+ (if(tighlyCoupled)"Tc" else "")) with InstructionAnticipatedPosition{
|
||||||
override def testParam = "IBUS=CACHED" + (if(compressed) " COMPRESSED=yes" else "")
|
override def testParam = "IBUS=CACHED" + (if(compressed) " COMPRESSED=yes" else "") + (if(tighlyCoupled)" IBUS_TC=yes" else "")
|
||||||
override def applyOn(config: VexRiscvConfig): Unit = config.plugins += new IBusCachedPlugin(
|
override def applyOn(config: VexRiscvConfig): Unit = {
|
||||||
|
val p = new IBusCachedPlugin(
|
||||||
resetVector = 0x80000000l,
|
resetVector = 0x80000000l,
|
||||||
compressedGen = compressed,
|
compressedGen = compressed,
|
||||||
prediction = prediction,
|
prediction = prediction,
|
||||||
|
@ -318,6 +321,9 @@ class IBusDimension extends VexRiscvDimension("IBus") {
|
||||||
twoCycleCache = twoCycleCache
|
twoCycleCache = twoCycleCache
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
if(tighlyCoupled) p.newTightlyCoupledPort(TightlyCoupledPortParameter("iBusTc", a => a(30 downto 28) === 0x0 && a(5)))
|
||||||
|
config.plugins += p
|
||||||
|
}
|
||||||
override def instructionAnticipatedOk() = !twoCycleCache || ((!twoCycleRam || wayCount == 1) && !compressed)
|
override def instructionAnticipatedOk() = !twoCycleCache || ((!twoCycleRam || wayCount == 1) && !compressed)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -523,8 +529,8 @@ class TestIndividualFeatures extends FunSuite {
|
||||||
|
|
||||||
|
|
||||||
// val testId = Some(mutable.HashSet[Int](0,28,45,93))
|
// val testId = Some(mutable.HashSet[Int](0,28,45,93))
|
||||||
// val testId = Some(mutable.HashSet[Int](5))
|
// val testId = Some(mutable.HashSet[Int](31))
|
||||||
// val seed = -2089952013329208578l
|
// val seed = -7716775349351274630l
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue