mirror of
https://github.com/enjoy-digital/litex.git
synced 2025-01-04 09:52:26 -05:00
245 lines
5.6 KiB
Python
245 lines
5.6 KiB
Python
from migen.fhdl.std import *
|
|
from migen.genlib.fsm import FSM, NextState
|
|
|
|
from lib.sata.common import *
|
|
|
|
def _encode_cmd(obj, layout, signal):
|
|
r = []
|
|
for k, v in sorted(layout.items()):
|
|
start = v.dword*32 + v.offset
|
|
end = start + v.width
|
|
if "_lsb" in k:
|
|
item = getattr(obj, k.replace("_lsb", ""))[:v.width]
|
|
elif "_msb" in k:
|
|
item = getattr(obj, k.replace("_msb", ""))[v.width:2*v.width]
|
|
else:
|
|
item = getattr(obj, k)
|
|
r.append(signal[start:end].eq(item))
|
|
return r
|
|
|
|
class SATATransportTX(Module):
|
|
def __init__(self, link):
|
|
self.sink = sink = Sink(transport_tx_layout(32))
|
|
|
|
###
|
|
|
|
cmd_ndwords = max(fis_reg_h2d_cmd_len, fis_data_cmd_len)
|
|
encoded_cmd = Signal(cmd_ndwords*32)
|
|
|
|
cnt = Signal(max=cmd_ndwords+1)
|
|
clr_cnt = Signal()
|
|
inc_cnt = Signal()
|
|
|
|
cmd_len = Signal(flen(cnt))
|
|
cmd_with_data = Signal()
|
|
|
|
cmd_send = Signal()
|
|
data_send = Signal()
|
|
cmd_done = Signal()
|
|
|
|
def test_type(name):
|
|
return sink.type == fis_types[name]
|
|
|
|
fsm = FSM(reset_state="IDLE")
|
|
self.submodules += fsm
|
|
|
|
fsm.act("IDLE",
|
|
clr_cnt.eq(1),
|
|
If(sink.stb & sink.sop,
|
|
If(test_type("REG_H2D"),
|
|
NextState("SEND_REG_H2D_CMD")
|
|
).Elif(test_type("DATA"),
|
|
NextState("SEND_DATA_CMD")
|
|
).Else(
|
|
sink.ack.eq(1)
|
|
)
|
|
).Else(
|
|
sink.ack.eq(1)
|
|
)
|
|
)
|
|
fsm.act("SEND_REG_H2D_CMD",
|
|
_encode_cmd(sink, fis_reg_h2d_layout, encoded_cmd),
|
|
cmd_len.eq(fis_reg_h2d_cmd_len-1),
|
|
cmd_send.eq(1),
|
|
If(cmd_done,
|
|
sink.ack.eq(1),
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
fsm.act("SEND_DATA_CMD",
|
|
_encode_cmd(sink, fis_data_layout, encoded_cmd),
|
|
cmd_len.eq(fis_data_cmd_len-1),
|
|
cmd_with_data.eq(1),
|
|
cmd_send.eq(1),
|
|
If(cmd_done,
|
|
NextState("SEND_DATA")
|
|
)
|
|
)
|
|
fsm.act("SEND_DATA",
|
|
data_send.eq(1),
|
|
sink.ack.eq(link.sink.ack),
|
|
If(sink.stb & sink.eop & sink.ack,
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
|
|
cmd_cases = {}
|
|
for i in range(cmd_ndwords):
|
|
cmd_cases[i] = [link.sink.d.eq(encoded_cmd[32*i:32*(i+1)])]
|
|
|
|
self.comb += \
|
|
If(cmd_send,
|
|
link.sink.stb.eq(1),
|
|
link.sink.sop.eq(cnt==0),
|
|
link.sink.eop.eq((cnt==cmd_len) & ~cmd_with_data),
|
|
Case(cnt, cmd_cases),
|
|
inc_cnt.eq(link.sink.ack),
|
|
cmd_done.eq((cnt==cmd_len) & link.sink.ack)
|
|
).Elif(data_send,
|
|
link.sink.stb.eq(sink.stb),
|
|
link.sink.sop.eq(0),
|
|
link.sink.eop.eq(sink.eop),
|
|
link.sink.d.eq(sink.data),
|
|
)
|
|
|
|
self.sync += \
|
|
If(clr_cnt,
|
|
cnt.eq(0)
|
|
).Elif(inc_cnt,
|
|
cnt.eq(cnt+1)
|
|
)
|
|
|
|
def _decode_cmd(signal, layout, obj):
|
|
r = []
|
|
for k, v in sorted(layout.items()):
|
|
start = v.dword*32+v.offset
|
|
end = start+v.width
|
|
if "_lsb" in k:
|
|
item = getattr(obj, k.replace("_lsb", ""))[:v.width]
|
|
elif "_msb" in k:
|
|
item = getattr(obj, k.replace("_msb", ""))[v.width:2*v.width]
|
|
else:
|
|
item = getattr(obj, k)
|
|
r.append(item.eq(signal[start:end]))
|
|
return r
|
|
|
|
class SATATransportRX(Module):
|
|
def __init__(self, link):
|
|
self.source = source = Source(transport_rx_layout(32))
|
|
|
|
###
|
|
|
|
cmd_ndwords = max(fis_reg_d2h_cmd_len, fis_dma_activate_d2h_cmd_len, fis_data_cmd_len)
|
|
encoded_cmd = Signal(cmd_ndwords*32)
|
|
|
|
cnt = Signal(max=cmd_ndwords+1)
|
|
clr_cnt = Signal()
|
|
inc_cnt = Signal()
|
|
|
|
cmd_len = Signal(flen(cnt))
|
|
|
|
cmd_receive = Signal()
|
|
data_receive = Signal()
|
|
cmd_done = Signal()
|
|
data_done = Signal()
|
|
|
|
def test_type(name):
|
|
return link.source.d[:8] == fis_types[name]
|
|
|
|
fsm = FSM(reset_state="IDLE")
|
|
self.submodules += fsm
|
|
|
|
fsm.act("IDLE",
|
|
clr_cnt.eq(1),
|
|
If(link.source.stb & link.source.sop,
|
|
If(test_type("REG_D2H"),
|
|
NextState("RECEIVE_REG_D2H_CMD")
|
|
).Elif(test_type("DMA_ACTIVATE_D2H"),
|
|
NextState("RECEIVE_DMA_ACTIVATE_D2H_CMD")
|
|
).Elif(test_type("DATA"),
|
|
NextState("RECEIVE_DATA_CMD"),
|
|
).Else(
|
|
# XXX: Better to ack?
|
|
link.source.ack.eq(1)
|
|
)
|
|
).Else(
|
|
link.source.ack.eq(1)
|
|
)
|
|
)
|
|
fsm.act("RECEIVE_REG_D2H_CMD",
|
|
cmd_len.eq(fis_reg_d2h_cmd_len-1),
|
|
cmd_receive.eq(1),
|
|
If(cmd_done,
|
|
NextState("PRESENT_REG_D2H_CMD")
|
|
)
|
|
)
|
|
fsm.act("PRESENT_REG_D2H_CMD",
|
|
source.stb.eq(1),
|
|
_decode_cmd(encoded_cmd, fis_reg_d2h_layout, source),
|
|
If(source.ack,
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
fsm.act("RECEIVE_DMA_ACTIVATE_D2H_CMD",
|
|
cmd_len.eq(fis_dma_activate_d2h_cmd_len-1),
|
|
cmd_receive.eq(1),
|
|
If(cmd_done,
|
|
NextState("PRESENT_DMA_ACTIVATE_D2H_CMD")
|
|
)
|
|
)
|
|
fsm.act("PRESENT_DMA_ACTIVATE_D2H_CMD",
|
|
source.stb.eq(1),
|
|
_decode_cmd(encoded_cmd, fis_dma_activate_d2h_layout, source),
|
|
If(source.ack,
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
fsm.act("RECEIVE_DATA_CMD",
|
|
cmd_len.eq(fis_data_cmd_len-1),
|
|
cmd_receive.eq(1),
|
|
If(cmd_done,
|
|
NextState("PRESENT_DATA")
|
|
)
|
|
)
|
|
fsm.act("PRESENT_DATA",
|
|
data_receive.eq(1),
|
|
source.stb.eq(link.source.stb),
|
|
_decode_cmd(encoded_cmd, fis_data_layout, source),
|
|
source.sop.eq(0), # XXX
|
|
source.eop.eq(link.source.eop),
|
|
source.data.eq(link.source.d),
|
|
If(source.stb & source.eop & source.ack,
|
|
NextState("IDLE")
|
|
)
|
|
)
|
|
|
|
cmd_cases = {}
|
|
for i in range(cmd_ndwords):
|
|
cmd_cases[i] = [encoded_cmd[32*i:32*(i+1)].eq(link.source.d)]
|
|
|
|
self.comb += \
|
|
If(cmd_receive,
|
|
If(link.source.stb,
|
|
inc_cnt.eq(1),
|
|
).Else(
|
|
inc_cnt.eq(0)
|
|
)
|
|
)
|
|
self.sync += \
|
|
If(cmd_receive,
|
|
Case(cnt, cmd_cases),
|
|
)
|
|
self.sync += \
|
|
If(clr_cnt,
|
|
cnt.eq(0)
|
|
).Elif(inc_cnt,
|
|
cnt.eq(cnt+1)
|
|
)
|
|
self.comb += cmd_done.eq((cnt==cmd_len) & link.source.ack)
|
|
self.comb += link.source.ack.eq(cmd_receive | (data_receive & source.ack))
|
|
|
|
class SATATransport(Module):
|
|
def __init__(self, link):
|
|
self.submodules.tx = SATATransportTX(link)
|
|
self.submodules.rx = SATATransportRX(link)
|
|
self.sink, self.source = self.tx.sink, self.rx.source
|