1
0
rocket-chip/rocket/src/main/scala/htif.scala

223 lines
7.6 KiB
Scala
Raw Normal View History

package rocket
import Chisel._
import Node._;
import Constants._;
class ioDebug(view: List[String] = null) extends Bundle(view)
{
val error_mode = Bool(OUTPUT);
}
class ioHost(w: Int, view: List[String] = null) extends Bundle(view)
{
2012-03-02 05:48:46 +01:00
val in = new ioDecoupled()(Bits(width = w)).flip
val out = new ioDecoupled()(Bits(width = w))
}
2012-03-25 00:23:29 +01:00
class PCRReq extends Bundle
{
val rw = Bool()
val addr = Bits(width = 5)
val data = Bits(width = 64)
}
class ioHTIF extends Bundle
{
val reset = Bool(INPUT)
val debug = new ioDebug
2012-03-25 00:23:29 +01:00
val pcr_req = (new ioDecoupled) { new PCRReq }.flip
val pcr_rep = (new ioPipe) { Bits(width = 64) }
}
class rocketHTIF(w: Int, ncores: Int, co: CoherencePolicyWithUncached) extends Component
{
val io = new Bundle {
val host = new ioHost(w)
2012-03-02 05:48:46 +01:00
val cpu = Vec(ncores) { new ioHTIF().flip }
val mem = new ioTileLink
}
val short_request_bits = 64
val long_request_bits = 576
require(short_request_bits % w == 0)
val rx_count_w = 13 + log2up(8) - log2up(w) // data size field is 12 bits
val rx_count = Reg(resetVal = UFix(0,rx_count_w))
val rx_shifter = Reg() { Bits(width = short_request_bits) }
val header = Reg() { Bits() }
val rx_shifter_in = Cat(io.host.in.bits, rx_shifter(short_request_bits-1,w))
when (io.host.in.valid && io.host.in.ready) {
rx_shifter := rx_shifter_in
rx_count := rx_count + UFix(1)
when (rx_count === UFix(short_request_bits/w-1)) {
header := rx_shifter_in
}
}
val rx_count_words = rx_count >> UFix(log2up(short_request_bits/w))
val packet_ram_wen = rx_count(log2up(short_request_bits/w)-1,0).andR &&
io.host.in.valid && io.host.in.ready
val packet_ram = Vec(long_request_bits/short_request_bits-1) { Reg() { Bits(width = short_request_bits) } }
when (packet_ram_wen) {
packet_ram(rx_count_words - UFix(1)) := rx_shifter_in
}
val cmd_readmem :: cmd_writemem :: cmd_readcr :: cmd_writecr :: cmd_ack :: cmd_nack :: Nil = Enum(6) { UFix() }
val cmd = header(3,0)
val size = header(15,4)
val seqno = header(23,16)
val addr = header(63,24).toUFix
val pcr_addr = addr(4,0)
val pcr_coreid = if (ncores == 1) UFix(0) else addr(20+log2up(ncores),20)
val pcr_wdata = packet_ram(0)
val nack = Mux(cmd === cmd_readmem || cmd === cmd_writemem, size != UFix((1 << OFFSET_BITS)/8),
Mux(cmd === cmd_readcr || cmd === cmd_writecr, size != UFix(1),
Bool(true)))
val tx_count = Reg(resetVal = UFix(0, log2up(long_request_bits/w+1)))
val packet_ram_raddr = (tx_count >> UFix(log2up(short_request_bits/w)))
when (io.host.out.valid && io.host.out.ready) {
tx_count := tx_count + UFix(1)
}
val rx_size = Mux(cmd === cmd_writemem || cmd === cmd_writecr, size, UFix(0))
val rx_done = rx_count >= UFix(short_request_bits/w) && rx_count_words-UFix(1) === rx_size
val tx_size = Mux(!nack && cmd === cmd_readmem, UFix((1 << OFFSET_BITS)/8),
Mux(!nack && cmd === cmd_readcr, UFix(1), UFix(0)))
val tx_done = packet_ram_raddr - UFix(1) === tx_size
2012-03-06 09:31:44 +01:00
val mem_acked = Reg(resetVal = Bool(false))
2012-03-07 00:47:19 +01:00
val mem_gxid = Reg() { Bits() }
val mem_needs_ack = Reg() { Bool() }
2012-03-06 09:31:44 +01:00
val mem_nacked = Reg(resetVal = Bool(false))
when (io.mem.xact_rep.valid) {
mem_acked := Bool(true)
mem_gxid := io.mem.xact_rep.bits.global_xact_id
mem_needs_ack := io.mem.xact_rep.bits.require_ack
}
2012-03-06 09:31:44 +01:00
when (io.mem.xact_abort.valid) { mem_nacked := Bool(true) }
2012-03-07 00:47:19 +01:00
val state_rx :: state_pcr :: state_mem_req :: state_mem_wdata :: state_mem_wresp :: state_mem_rdata :: state_mem_finish :: state_tx :: Nil = Enum(8) { UFix() }
val state = Reg(resetVal = state_rx)
when (state === state_rx && rx_done) {
state := Mux(cmd === cmd_readmem || cmd === cmd_writemem, state_mem_req,
Mux(cmd === cmd_readcr || cmd === cmd_writecr, state_pcr,
state_tx))
}
val pcr_done = Reg() { Bool() }
when (state === state_pcr && pcr_done) {
state := state_tx
}
val mem_cnt = Reg(resetVal = UFix(0, log2up(REFILL_CYCLES)))
when (state === state_mem_req && io.mem.xact_init.ready) {
state := Mux(cmd === cmd_writemem, state_mem_wdata, state_mem_rdata)
}
2012-03-06 09:31:44 +01:00
when (state === state_mem_wdata && io.mem.xact_init_data.ready) {
when (mem_cnt.andR) {
2012-03-06 09:31:44 +01:00
state := state_mem_wresp
}
mem_cnt := mem_cnt + UFix(1)
}
2012-03-06 09:31:44 +01:00
when (state === state_mem_wresp) {
when (mem_nacked) {
state := state_mem_req
mem_nacked := Bool(false)
}
when (mem_acked) {
2012-03-07 00:47:19 +01:00
state := state_mem_finish
2012-03-06 09:31:44 +01:00
mem_acked := Bool(false)
}
}
when (state === state_mem_rdata) {
when (mem_nacked) {
state := state_mem_req
mem_nacked := Bool(false)
}
when (io.mem.xact_rep.valid) {
when (mem_cnt.andR) {
2012-03-07 00:47:19 +01:00
state := state_mem_finish
2012-03-06 09:31:44 +01:00
}
mem_cnt := mem_cnt + UFix(1)
}
mem_acked := Bool(false)
}
2012-03-07 00:47:19 +01:00
when (state === state_mem_finish && io.mem.xact_finish.ready) {
state := state_tx
}
when (state === state_tx && tx_done) {
rx_count := UFix(0)
tx_count := UFix(0)
state := state_rx
}
var mem_req_data: Bits = null
for (i <- 0 until MEM_DATA_BITS/short_request_bits) {
val idx = Cat(mem_cnt, UFix(i, log2up(MEM_DATA_BITS/short_request_bits)))
when (state === state_mem_rdata && io.mem.xact_rep.valid) {
packet_ram(idx) := io.mem.xact_rep.bits.data((i+1)*short_request_bits-1, i*short_request_bits)
}
mem_req_data = Cat(packet_ram(idx), mem_req_data)
}
io.mem.xact_init.valid := state === state_mem_req
io.mem.xact_init.bits.x_type := Mux(cmd === cmd_writemem, co.getTransactionInitTypeOnUncachedWrite, co.getTransactionInitTypeOnUncachedRead)
io.mem.xact_init.bits.address := addr >> UFix(OFFSET_BITS-3)
io.mem.xact_init_data.valid:= state === state_mem_wdata
io.mem.xact_init_data.bits.data := mem_req_data
io.mem.xact_finish.valid := (state === state_mem_finish) && mem_needs_ack
2012-03-07 00:47:19 +01:00
io.mem.xact_finish.bits.global_xact_id := mem_gxid
2012-03-14 03:10:54 +01:00
val probe_q = (new queue(1)) { new ProbeReply }
2012-03-14 00:56:47 +01:00
probe_q.io.enq.valid := io.mem.probe_req.valid
io.mem.probe_req.ready := probe_q.io.enq.ready
probe_q.io.enq.bits := co.newProbeReply(io.mem.probe_req.bits, co.newStateOnFlush())
2012-03-14 00:56:47 +01:00
io.mem.probe_rep <> probe_q.io.deq
io.mem.probe_rep_data.valid := Bool(false)
pcr_done := Bool(false)
val pcr_mux = (new Mux1H(ncores)) { Bits(width = 64) }
for (i <- 0 until ncores) {
2012-03-25 00:23:29 +01:00
val my_reset = Reg(resetVal = Bool(true))
val rdata = Reg() { Bits() }
val cpu = io.cpu(i)
val me = pcr_coreid === UFix(i)
2012-03-25 00:23:29 +01:00
cpu.pcr_req.valid := state === state_pcr && me
cpu.pcr_req.bits.rw := cmd === cmd_writecr
cpu.pcr_req.bits.addr := pcr_addr
cpu.pcr_req.bits.data := pcr_wdata
cpu.reset := my_reset
when (state === state_pcr && me && cmd === cmd_writecr) {
pcr_done := cpu.pcr_req.ready
when (pcr_addr === PCR_RESET) {
my_reset := pcr_wdata(0)
2012-03-25 00:23:29 +01:00
}
}
2012-03-25 00:23:29 +01:00
when (cpu.pcr_rep.valid) {
pcr_done := Bool(true)
2012-03-25 00:23:29 +01:00
rdata := cpu.pcr_rep.bits
}
2012-03-25 00:23:29 +01:00
pcr_mux.io.sel(i) := me
pcr_mux.io.in(i) := Mux(pcr_addr === PCR_RESET, my_reset, rdata)
}
val tx_cmd = Mux(nack, cmd_nack, cmd_ack)
val tx_cmd_ext = Cat(Bits(0, 4-tx_cmd.getWidth), tx_cmd)
val tx_size_ext = Cat(Bits(0, 12-tx_size.getWidth), tx_size)
val tx_header = Cat(addr, seqno, tx_size_ext, tx_cmd_ext)
val tx_data = Mux(packet_ram_raddr === UFix(0), tx_header,
Mux(packet_ram_raddr === UFix(1) && cmd === cmd_readcr, pcr_mux.io.out,
packet_ram(packet_ram_raddr - UFix(1))))
io.host.in.ready := state === state_rx && !rx_done
io.host.out.valid := state === state_tx && !tx_done
io.host.out.bits := tx_data >> Cat(tx_count(log2up(short_request_bits/w)-1,0), Bits(0, log2up(w)))
}