1
0
rocket-chip/uncore/src/main/scala/memserdes.scala

187 lines
5.6 KiB
Scala
Raw Normal View History

package uncore
2012-03-26 02:03:58 +02:00
import Chisel._
import scala.math._
2013-08-12 19:36:44 +02:00
trait HasMemData extends Bundle {
val data = Bits(width = MEM_DATA_BITS)
}
2013-08-12 19:36:44 +02:00
trait HasMemAddr extends Bundle {
val addr = UInt(width = MEM_ADDR_BITS)
}
2013-08-12 19:36:44 +02:00
trait HasMemTag extends Bundle {
val tag = UInt(width = MEM_TAG_BITS)
}
2013-08-12 19:36:44 +02:00
class MemReqCmd extends HasMemAddr with HasMemTag {
val rw = Bool()
}
class MemResp extends HasMemData with HasMemTag
class MemData extends HasMemData
class ioMem extends Bundle {
2013-08-12 19:36:44 +02:00
val req_cmd = Decoupled(new MemReqCmd)
val req_data = Decoupled(new MemData)
val resp = Decoupled(new MemResp).flip
}
class ioMemPipe extends Bundle {
2013-08-12 19:36:44 +02:00
val req_cmd = Decoupled(new MemReqCmd)
val req_data = Decoupled(new MemData)
val resp = Valid(new MemResp).flip
}
class ioMemSerialized(w: Int) extends Bundle
2012-03-26 02:03:58 +02:00
{
2013-08-12 19:36:44 +02:00
val req = Decoupled(Bits(width = w))
val resp = Valid(Bits(width = w)).flip
2012-03-26 02:03:58 +02:00
}
2013-08-12 19:36:44 +02:00
class MemSerdes(w: Int) extends Module
2012-03-26 02:03:58 +02:00
{
val io = new Bundle {
val wide = new ioMem().flip
val narrow = new ioMemSerialized(w)
2012-03-26 02:03:58 +02:00
}
val abits = io.wide.req_cmd.bits.toBits.getWidth
val dbits = io.wide.req_data.bits.toBits.getWidth
val rbits = io.wide.resp.bits.getWidth
2013-08-12 19:36:44 +02:00
val out_buf = Reg(Bits())
val in_buf = Reg(Bits())
2012-03-26 02:03:58 +02:00
2013-08-12 19:36:44 +02:00
val s_idle :: s_read_addr :: s_write_addr :: s_write_idle :: s_write_data :: Nil = Enum(5) { UInt() }
2013-08-16 00:27:38 +02:00
val state = Reg(init=s_idle)
val send_cnt = Reg(init=UInt(0, log2Up((max(abits, dbits)+w-1)/w)))
val data_send_cnt = Reg(init=UInt(0, log2Up(REFILL_CYCLES)))
2013-08-12 19:36:44 +02:00
val adone = io.narrow.req.ready && send_cnt === UInt((abits-1)/w)
val ddone = io.narrow.req.ready && send_cnt === UInt((dbits-1)/w)
2012-03-26 02:03:58 +02:00
2012-03-26 08:03:20 +02:00
when (io.narrow.req.valid && io.narrow.req.ready) {
2013-08-12 19:36:44 +02:00
send_cnt := send_cnt + UInt(1)
out_buf := out_buf >> UInt(w)
2012-03-26 08:03:20 +02:00
}
when (io.wide.req_cmd.valid && io.wide.req_cmd.ready) {
out_buf := io.wide.req_cmd.bits.toBits
}
when (io.wide.req_data.valid && io.wide.req_data.ready) {
out_buf := io.wide.req_data.bits.toBits
}
io.wide.req_cmd.ready := state === s_idle
io.wide.req_data.ready := state === s_write_idle
io.narrow.req.valid := state === s_read_addr || state === s_write_addr || state === s_write_data
io.narrow.req.bits := out_buf
when (state === s_idle && io.wide.req_cmd.valid) {
state := Mux(io.wide.req_cmd.bits.rw, s_write_addr, s_read_addr)
2012-03-26 02:03:58 +02:00
}
when (state === s_read_addr && adone) {
state := s_idle
2013-08-12 19:36:44 +02:00
send_cnt := UInt(0)
2012-03-26 02:03:58 +02:00
}
when (state === s_write_addr && adone) {
state := s_write_idle
2013-08-12 19:36:44 +02:00
send_cnt := UInt(0)
2012-03-26 02:03:58 +02:00
}
when (state === s_write_idle && io.wide.req_data.valid) {
state := s_write_data
}
when (state === s_write_data && ddone) {
2013-08-12 19:36:44 +02:00
data_send_cnt := data_send_cnt + UInt(1)
state := Mux(data_send_cnt === UInt(REFILL_CYCLES-1), s_idle, s_write_idle)
send_cnt := UInt(0)
2012-03-26 02:03:58 +02:00
}
2013-08-16 00:27:38 +02:00
val recv_cnt = Reg(init=UInt(0, log2Up((rbits+w-1)/w)))
val data_recv_cnt = Reg(init=UInt(0, log2Up(REFILL_CYCLES)))
val resp_val = Reg(init=Bool(false))
2012-03-26 02:03:58 +02:00
resp_val := Bool(false)
when (io.narrow.resp.valid) {
2013-08-12 19:36:44 +02:00
recv_cnt := recv_cnt + UInt(1)
when (recv_cnt === UInt((rbits-1)/w)) {
recv_cnt := UInt(0)
data_recv_cnt := data_recv_cnt + UInt(1)
2012-03-26 02:03:58 +02:00
resp_val := Bool(true)
}
in_buf := Cat(io.narrow.resp.bits, in_buf((rbits+w-1)/w*w-1,w))
2012-03-26 02:03:58 +02:00
}
io.wide.resp.valid := resp_val
2013-08-25 00:24:17 +02:00
io.wide.resp.bits := io.wide.resp.bits.fromBits(in_buf)
2012-03-26 02:03:58 +02:00
}
2012-03-26 08:03:20 +02:00
class MemDesserIO(w: Int) extends Bundle {
val narrow = new ioMemSerialized(w).flip
val wide = new ioMem
}
2013-08-12 19:36:44 +02:00
class MemDesser(w: Int) extends Module // test rig side
2012-03-26 08:03:20 +02:00
{
val io = new MemDesserIO(w)
2012-03-26 08:03:20 +02:00
val abits = io.wide.req_cmd.bits.toBits.getWidth
val dbits = io.wide.req_data.bits.toBits.getWidth
val rbits = io.wide.resp.bits.getWidth
require(dbits >= abits && rbits >= dbits)
2013-08-16 00:27:38 +02:00
val recv_cnt = Reg(init=UInt(0, log2Up((rbits+w-1)/w)))
val data_recv_cnt = Reg(init=UInt(0, log2Up(REFILL_CYCLES)))
2013-08-12 19:36:44 +02:00
val adone = io.narrow.req.valid && recv_cnt === UInt((abits-1)/w)
val ddone = io.narrow.req.valid && recv_cnt === UInt((dbits-1)/w)
val rdone = io.narrow.resp.valid && recv_cnt === UInt((rbits-1)/w)
2012-03-26 08:03:20 +02:00
2013-08-12 19:36:44 +02:00
val s_cmd_recv :: s_cmd :: s_data_recv :: s_data :: s_reply :: Nil = Enum(5) { UInt() }
2013-08-16 00:27:38 +02:00
val state = Reg(init=s_cmd_recv)
2012-03-26 08:03:20 +02:00
2013-08-12 19:36:44 +02:00
val in_buf = Reg(Bits())
2012-03-26 08:03:20 +02:00
when (io.narrow.req.valid && io.narrow.req.ready || io.narrow.resp.valid) {
2013-08-12 19:36:44 +02:00
recv_cnt := recv_cnt + UInt(1)
in_buf := Cat(io.narrow.req.bits, in_buf((rbits+w-1)/w*w-1,w))
2012-03-26 08:03:20 +02:00
}
io.narrow.req.ready := state === s_cmd_recv || state === s_data_recv
when (state === s_cmd_recv && adone) {
state := s_cmd
2013-08-12 19:36:44 +02:00
recv_cnt := UInt(0)
2012-03-26 08:03:20 +02:00
}
when (state === s_cmd && io.wide.req_cmd.ready) {
state := Mux(io.wide.req_cmd.bits.rw, s_data_recv, s_reply)
}
when (state === s_data_recv && ddone) {
state := s_data
2013-08-12 19:36:44 +02:00
recv_cnt := UInt(0)
2012-03-26 08:03:20 +02:00
}
when (state === s_data && io.wide.req_data.ready) {
state := s_data_recv
2013-08-12 19:36:44 +02:00
when (data_recv_cnt === UInt(REFILL_CYCLES-1)) {
2012-03-26 08:03:20 +02:00
state := s_cmd_recv
}
2013-08-12 19:36:44 +02:00
data_recv_cnt := data_recv_cnt + UInt(1)
2012-03-26 08:03:20 +02:00
}
when (rdone) { // state === s_reply
2013-08-12 19:36:44 +02:00
when (data_recv_cnt === UInt(REFILL_CYCLES-1)) {
2012-03-26 08:03:20 +02:00
state := s_cmd_recv
}
2013-08-12 19:36:44 +02:00
recv_cnt := UInt(0)
data_recv_cnt := data_recv_cnt + UInt(1)
2012-03-26 08:03:20 +02:00
}
2013-08-12 19:36:44 +02:00
val req_cmd = in_buf >> UInt(((rbits+w-1)/w - (abits+w-1)/w)*w)
2012-03-26 08:03:20 +02:00
io.wide.req_cmd.valid := state === s_cmd
io.wide.req_cmd.bits := io.wide.req_cmd.bits.fromBits(req_cmd)
2012-03-26 08:03:20 +02:00
io.wide.req_data.valid := state === s_data
2013-08-12 19:36:44 +02:00
io.wide.req_data.bits.data := in_buf >> UInt(((rbits+w-1)/w - (dbits+w-1)/w)*w)
2012-03-26 08:03:20 +02:00
2013-08-12 19:36:44 +02:00
val dataq = Module(new Queue(new MemResp, REFILL_CYCLES))
2012-03-26 08:03:20 +02:00
dataq.io.enq <> io.wide.resp
2013-08-12 19:36:44 +02:00
dataq.io.deq.ready := recv_cnt === UInt((rbits-1)/w)
2012-03-26 08:03:20 +02:00
io.narrow.resp.valid := dataq.io.deq.valid
2013-08-12 19:36:44 +02:00
io.narrow.resp.bits := dataq.io.deq.bits.toBits >> (recv_cnt * UInt(w))
2012-03-26 08:03:20 +02:00
}