blob: 6c9c47e5d707c13fbdacf048bdca500c247532a3 [file] [log] [blame]
// 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.
// Scalar Core Frontend
package kelvin
import chisel3._
import chisel3.util._
import common._
import _root_.circt.stage.ChiselStage
object SCore {
def apply(p: Parameters): SCore = {
return Module(new SCore(p))
}
}
class SCore(p: Parameters) extends Module {
val io = IO(new Bundle {
val csr = new CsrInOutIO(p)
val halted = Output(Bool())
val fault = Output(Bool())
val ibus = new IBusIO(p)
val dbus = new DBusIO(p)
val ubus = new DBusIO(p)
val vldst = if (p.enableVector) { Some(Output(Bool())) } else { None }
val vcore = if (p.enableVector) {
Some(Flipped(new VCoreIO(p)))
} else { None }
val iflush = new IFlushIO(p)
val dflush = new DFlushIO(p)
val slog = new SLogIO(p)
val debug = new DebugIO(p)
})
// The functional units that make up the core.
val regfile = Regfile(p)
val fetch = Fetch(p)
val decode = (0 until p.instructionLanes).map(x => Seq(Decode(p, x))).reduce(_ ++ _)
val alu = Seq.fill(p.instructionLanes)(Alu(p))
val bru = Seq.fill(p.instructionLanes)(Bru(p))
val csr = Csr(p)
val lsu = Lsu(p)
val mlu = Mlu(p)
val dvu = Dvu(p)
// Wire up the core.
val branchTaken = bru.map(x => x.io.taken.valid).reduce(_||_)
// ---------------------------------------------------------------------------
// IFlush
val iflush = RegInit(false.B)
when (bru(0).io.iflush) {
iflush := true.B
} .elsewhen (fetch.io.iflush.ready && io.iflush.ready &&
lsu.io.flush.ready && lsu.io.flush.fencei) {
iflush := false.B
}
io.dflush.valid := lsu.io.flush.valid
io.dflush.all := lsu.io.flush.all
io.dflush.clean := lsu.io.flush.clean
lsu.io.flush.ready := io.dflush.ready
for (i <- 1 until p.instructionLanes) {
assert(!bru(i).io.iflush)
}
// ---------------------------------------------------------------------------
// Fetch
fetch.io.csr := io.csr.in
for (i <- 0 until p.instructionLanes) {
fetch.io.branch(i) := bru(i).io.taken
}
fetch.io.linkPort := regfile.io.linkPort
fetch.io.iflush.valid := iflush
// ---------------------------------------------------------------------------
// Decode
val mask = VecInit(decode.map(_.io.inst.ready).scan(true.B)(_ && _))
for (i <- 0 until p.instructionLanes) {
decode(i).io.inst.valid := fetch.io.inst.lanes(i).valid && mask(i)
fetch.io.inst.lanes(i).ready := decode(i).io.inst.ready && mask(i)
decode(i).io.inst.addr := fetch.io.inst.lanes(i).addr
decode(i).io.inst.inst := fetch.io.inst.lanes(i).inst
decode(i).io.inst.brchFwd := fetch.io.inst.lanes(i).brchFwd
decode(i).io.branchTaken := branchTaken
decode(i).io.halted := csr.io.halted
}
// Interlock based on regfile write port dependencies.
decode(0).io.interlock := bru(0).io.interlock
for (i <- 1 until p.instructionLanes) {
decode(i).io.interlock := decode(i - 1).io.interlock
}
// Serialize opcodes with only one pipeline.
decode(0).io.serializeIn.defaults()
for (i <- 1 until p.instructionLanes) {
decode(i).io.serializeIn := decode(i - 1).io.serializeOut
}
// In decode update multi-issue scoreboard state.
val scoreboard_spec = decode.map(_.io.scoreboard.spec).scan(0.U)(_|_)
for (i <- 0 until p.instructionLanes) {
decode(i).io.scoreboard.comb := regfile.io.scoreboard.comb | scoreboard_spec(i)
decode(i).io.scoreboard.regd := regfile.io.scoreboard.regd | scoreboard_spec(i)
}
decode(0).io.mactive := (if (p.enableVector) { io.vcore.get.mactive } else { false.B })
for (i <- 1 until p.instructionLanes) {
decode(i).io.mactive := false.B
}
// ---------------------------------------------------------------------------
// ALU
for (i <- 0 until p.instructionLanes) {
alu(i).io.req := decode(i).io.alu
alu(i).io.rs1 := regfile.io.readData(2 * i + 0)
alu(i).io.rs2 := regfile.io.readData(2 * i + 1)
}
// ---------------------------------------------------------------------------
// Branch Unit
for (i <- 0 until p.instructionLanes) {
bru(i).io.req := decode(i).io.bru
bru(i).io.rs1 := regfile.io.readData(2 * i + 0)
bru(i).io.rs2 := regfile.io.readData(2 * i + 1)
bru(i).io.target := regfile.io.target(i)
}
bru(0).io.csr <> csr.io.bru
for (i <- 1 until p.instructionLanes) {
bru(i).io.csr.defaults()
}
io.iflush.valid := iflush
// Instruction counters
csr.io.counters.rfwriteCount := regfile.io.rfwriteCount
csr.io.counters.storeCount := lsu.io.storeCount
csr.io.counters.branchCount := bru(0).io.taken.valid
if (p.enableVector) {
csr.io.counters.vrfwriteCount.get := io.vcore.get.vrfwriteCount
csr.io.counters.vstoreCount.get := io.vcore.get.vstoreCount
}
// ---------------------------------------------------------------------------
// Control Status Unit
csr.io.csr <> io.csr
csr.io.req <> decode(0).io.csr
csr.io.rs1 := regfile.io.readData(0)
if (p.enableVector) {
csr.io.vcore.get.undef := io.vcore.get.undef
}
// ---------------------------------------------------------------------------
// Status
io.halted := csr.io.halted
io.fault := csr.io.fault
// ---------------------------------------------------------------------------
// Load/Store Unit
lsu.io.busPort := regfile.io.busPort
lsu.io.req <> decode.map(_.io.lsu)
// ---------------------------------------------------------------------------
// Multiplier Unit
for (i <- 0 until p.instructionLanes) {
mlu.io.req(i) := decode(i).io.mlu
mlu.io.rs1(i) := regfile.io.readData(2 * i)
mlu.io.rs2(i) := regfile.io.readData((2 * i) + 1)
}
// ---------------------------------------------------------------------------
// Divide Unit
dvu.io.req <> decode(0).io.dvu
dvu.io.rs1 := regfile.io.readData(0)
dvu.io.rs2 := regfile.io.readData(1)
dvu.io.rd.ready := !mlu.io.rd.valid
// TODO: make port conditional on pipeline index.
for (i <- 1 until p.instructionLanes) {
decode(i).io.dvu.ready := false.B
}
// ---------------------------------------------------------------------------
// Register File
for (i <- 0 until p.instructionLanes) {
regfile.io.readAddr(2 * i + 0) := decode(i).io.rs1Read
regfile.io.readAddr(2 * i + 1) := decode(i).io.rs2Read
regfile.io.readSet(2 * i + 0) := decode(i).io.rs1Set
regfile.io.readSet(2 * i + 1) := decode(i).io.rs2Set
regfile.io.writeAddr(i) := decode(i).io.rdMark
regfile.io.busAddr(i) := decode(i).io.busRead
val csr0Valid = if (i == 0) csr.io.rd.valid else false.B
val csr0Addr = if (i == 0) csr.io.rd.addr else 0.U
val csr0Data = if (i == 0) csr.io.rd.data else 0.U
regfile.io.writeData(i).valid := csr0Valid ||
alu(i).io.rd.valid || bru(i).io.rd.valid ||
(if (p.enableVector) {
io.vcore.get.rd(i).valid
} else { false.B })
regfile.io.writeData(i).addr :=
MuxOR(csr0Valid, csr0Addr) |
MuxOR(alu(i).io.rd.valid, alu(i).io.rd.addr) |
MuxOR(bru(i).io.rd.valid, bru(i).io.rd.addr) |
(if (p.enableVector) {
MuxOR(io.vcore.get.rd(i).valid, io.vcore.get.rd(i).addr)
} else { false.B })
regfile.io.writeData(i).data :=
MuxOR(csr0Valid, csr0Data) |
MuxOR(alu(i).io.rd.valid, alu(i).io.rd.data) |
MuxOR(bru(i).io.rd.valid, bru(i).io.rd.data) |
(if (p.enableVector) {
MuxOR(io.vcore.get.rd(i).valid, io.vcore.get.rd(i).data)
} else { false.B })
if (p.enableVector) {
assert((csr0Valid +&
alu(i).io.rd.valid +& bru(i).io.rd.valid +&
io.vcore.get.rd(i).valid) <= 1.U)
} else {
assert((csr0Valid +&
alu(i).io.rd.valid +& bru(i).io.rd.valid) <= 1.U)
}
}
val mluDvuOffset = p.instructionLanes
regfile.io.writeData(mluDvuOffset).valid := mlu.io.rd.valid || dvu.io.rd.valid
regfile.io.writeData(mluDvuOffset).addr := Mux(mlu.io.rd.valid, mlu.io.rd.addr, dvu.io.rd.addr)
regfile.io.writeData(mluDvuOffset).data := Mux(mlu.io.rd.valid, mlu.io.rd.data, dvu.io.rd.data)
assert(!(mlu.io.rd.valid && (dvu.io.rd.valid && dvu.io.rd.ready))) // TODO: stall dvu on mlu write
val lsuOffset = p.instructionLanes + 1
regfile.io.writeData(lsuOffset).valid := lsu.io.rd.valid
regfile.io.writeData(lsuOffset).addr := lsu.io.rd.addr
regfile.io.writeData(lsuOffset).data := lsu.io.rd.data
val writeMask = bru.map(_.io.taken.valid).scan(false.B)(_||_)
for (i <- 0 until p.instructionLanes) {
regfile.io.writeMask(i).valid := writeMask(i)
}
// ---------------------------------------------------------------------------
// Vector Extension
if (p.enableVector) {
io.vcore.get.vinst <> decode.map(_.io.vinst.get)
io.vcore.get.rs := regfile.io.readData
}
// ---------------------------------------------------------------------------
// Fetch Bus
io.ibus <> fetch.io.ibus
// ---------------------------------------------------------------------------
// Local Data Bus Port
io.dbus <> lsu.io.dbus
io.ubus <> lsu.io.ubus
if (p.enableVector) {
io.vldst.get := lsu.io.vldst
}
// ---------------------------------------------------------------------------
// Scalar logging interface
val slogValid = RegInit(false.B)
val slogAddr = Reg(UInt(2.W))
val slogEn = decode(0).io.slog
slogValid := slogEn
when (slogEn) {
slogAddr := decode(0).io.inst.inst(14,12)
}
io.slog.valid := slogValid
io.slog.addr := MuxOR(slogValid, slogAddr)
io.slog.data := MuxOR(slogValid, regfile.io.readData(0).data)
// ---------------------------------------------------------------------------
// DEBUG
val cycles = RegInit(0.U(32.W))
cycles := cycles + 1.U
io.debug.cycles := cycles
val debugEn = RegInit(0.U(p.instructionLanes.W))
val debugAddr = Reg(Vec(p.instructionLanes, UInt(32.W)))
val debugInst = Reg(Vec(p.instructionLanes, UInt(32.W)))
val debugBrch = Cat(bru.map(_.io.taken.valid).scanRight(false.B)(_ || _))
debugEn := Cat(fetch.io.inst.lanes.map(x => x.valid && x.ready && !branchTaken))
for (i <- 0 until p.instructionLanes) {
debugAddr(i) := fetch.io.inst.lanes(i).addr
debugInst(i) := fetch.io.inst.lanes(i).inst
}
io.debug.en := debugEn & ~debugBrch
io.debug.addr <> debugAddr
io.debug.inst <> debugInst
}
object EmitSCore extends App {
val p = new Parameters
ChiselStage.emitSystemVerilogFile(new SCore(p), args)
}