2015-01-16 17:52:41 -05:00
|
|
|
from litesata.common import *
|
|
|
|
from litesata.core.link.crc import LiteSATACRCInserter, LiteSATACRCChecker
|
|
|
|
from litesata.core.link.scrambler import LiteSATAScrambler
|
|
|
|
from litesata.core.link.cont import LiteSATACONTInserter, LiteSATACONTRemover
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-03 09:29:01 -05:00
|
|
|
from_rx = [
|
|
|
|
("idle", 1),
|
|
|
|
("insert", 32),
|
|
|
|
("det", 32)
|
|
|
|
]
|
|
|
|
|
2015-01-16 17:52:41 -05:00
|
|
|
class LiteSATALinkTX(Module):
|
2014-12-23 12:26:07 -05:00
|
|
|
def __init__(self, phy):
|
2014-12-14 04:52:56 -05:00
|
|
|
self.sink = Sink(link_description(32))
|
2014-12-03 09:29:01 -05:00
|
|
|
self.from_rx = Sink(from_rx)
|
|
|
|
|
|
|
|
###
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-19 16:50:35 -05:00
|
|
|
self.fsm = fsm = FSM(reset_state="IDLE")
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-11-11 06:26:32 -05:00
|
|
|
# insert CRC
|
2015-01-16 17:52:41 -05:00
|
|
|
crc = LiteSATACRCInserter(link_description(32))
|
2014-12-03 09:29:01 -05:00
|
|
|
self.submodules += crc
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-11-11 06:26:32 -05:00
|
|
|
# scramble
|
2015-01-16 17:52:41 -05:00
|
|
|
scrambler = LiteSATAScrambler(link_description(32))
|
2014-12-03 09:29:01 -05:00
|
|
|
self.submodules += scrambler
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-03 09:29:01 -05:00
|
|
|
# connect CRC / scrambler
|
2014-11-11 06:26:32 -05:00
|
|
|
self.comb += [
|
2014-12-03 09:29:01 -05:00
|
|
|
Record.connect(self.sink, crc.sink),
|
|
|
|
Record.connect(crc.source, scrambler.sink)
|
2014-11-11 06:26:32 -05:00
|
|
|
]
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-03 05:12:26 -05:00
|
|
|
# inserter CONT and scrambled data between
|
|
|
|
# CONT and next primitive
|
2015-01-16 17:52:41 -05:00
|
|
|
self.cont = cont = BufferizeEndpoints(LiteSATACONTInserter(phy_description(32)), "source")
|
2014-12-03 05:12:26 -05:00
|
|
|
|
2014-11-11 06:26:32 -05:00
|
|
|
# datas / primitives mux
|
2014-12-03 09:29:01 -05:00
|
|
|
insert = Signal(32)
|
2014-11-11 06:26:32 -05:00
|
|
|
self.comb += [
|
2014-12-03 09:29:01 -05:00
|
|
|
If(self.from_rx.insert,
|
|
|
|
cont.sink.stb.eq(1),
|
|
|
|
cont.sink.data.eq(self.from_rx.insert),
|
|
|
|
cont.sink.charisk.eq(0x0001),
|
|
|
|
).
|
|
|
|
Elif(insert,
|
|
|
|
cont.sink.stb.eq(1),
|
|
|
|
cont.sink.data.eq(insert),
|
|
|
|
cont.sink.charisk.eq(0x0001),
|
|
|
|
).Elif(fsm.ongoing("COPY"),
|
|
|
|
cont.sink.stb.eq(scrambler.source.stb),
|
|
|
|
cont.sink.data.eq(scrambler.source.d),
|
|
|
|
scrambler.source.ack.eq(cont.sink.ack),
|
|
|
|
cont.sink.charisk.eq(0)
|
2014-11-11 06:26:32 -05:00
|
|
|
)
|
|
|
|
]
|
2014-12-03 09:29:01 -05:00
|
|
|
self.comb += Record.connect(cont.source, phy.sink)
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-03 09:29:01 -05:00
|
|
|
# FSM
|
|
|
|
fsm.act("IDLE",
|
2014-12-05 14:26:09 -05:00
|
|
|
scrambler.reset.eq(1),
|
|
|
|
If(self.from_rx.idle,
|
|
|
|
insert.eq(primitives["SYNC"]),
|
|
|
|
If(scrambler.source.stb & scrambler.source.sop,
|
2014-12-19 17:10:51 -05:00
|
|
|
If(self.from_rx.det == primitives["SYNC"],
|
|
|
|
NextState("RDY")
|
|
|
|
)
|
2014-12-03 09:29:01 -05:00
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("RDY",
|
|
|
|
insert.eq(primitives["X_RDY"]),
|
2014-12-05 14:26:09 -05:00
|
|
|
If(~self.from_rx.idle,
|
|
|
|
NextState("IDLE")
|
|
|
|
).Elif(self.from_rx.det == primitives["R_RDY"],
|
2014-12-03 09:29:01 -05:00
|
|
|
NextState("SOF")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("SOF",
|
|
|
|
insert.eq(primitives["SOF"]),
|
|
|
|
If(phy.sink.ack,
|
|
|
|
NextState("COPY")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("COPY",
|
|
|
|
If(self.from_rx.det == primitives["HOLD"],
|
|
|
|
insert.eq(primitives["HOLDA"]),
|
|
|
|
).Elif(~scrambler.source.stb,
|
|
|
|
insert.eq(primitives["HOLD"]),
|
|
|
|
).Elif(scrambler.source.stb & scrambler.source.eop & scrambler.source.ack,
|
|
|
|
NextState("EOF")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("EOF",
|
|
|
|
insert.eq(primitives["EOF"]),
|
|
|
|
If(phy.sink.ack,
|
|
|
|
NextState("WTRM")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("WTRM",
|
|
|
|
insert.eq(primitives["WTRM"]),
|
|
|
|
If(self.from_rx.det == primitives["R_OK"],
|
|
|
|
NextState("IDLE")
|
|
|
|
).Elif(self.from_rx.det == primitives["R_ERR"],
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
)
|
2014-12-03 05:12:26 -05:00
|
|
|
|
2015-01-16 17:52:41 -05:00
|
|
|
class LiteSATALinkRX(Module):
|
2014-12-03 09:29:01 -05:00
|
|
|
def __init__(self, phy):
|
2014-12-14 04:52:56 -05:00
|
|
|
self.source = Source(link_description(32))
|
2014-12-03 09:29:01 -05:00
|
|
|
self.to_tx = Source(from_rx)
|
2014-12-03 05:50:31 -05:00
|
|
|
|
2014-12-03 09:29:01 -05:00
|
|
|
###
|
2014-12-03 05:50:31 -05:00
|
|
|
|
2014-12-19 16:50:35 -05:00
|
|
|
self.fsm = fsm = FSM(reset_state="IDLE")
|
2014-12-03 09:29:01 -05:00
|
|
|
|
|
|
|
# CONT remover
|
2015-01-16 17:52:41 -05:00
|
|
|
self.cont = cont = BufferizeEndpoints(LiteSATACONTRemover(phy_description(32)), "source")
|
2014-12-03 09:29:01 -05:00
|
|
|
self.comb += Record.connect(phy.source, cont.sink)
|
2014-12-03 05:50:31 -05:00
|
|
|
|
2014-11-11 06:26:32 -05:00
|
|
|
# datas / primitives detection
|
2014-12-03 09:29:01 -05:00
|
|
|
insert = Signal(32)
|
|
|
|
det = Signal(32)
|
2014-11-11 06:26:32 -05:00
|
|
|
self.comb += \
|
2014-12-03 09:29:01 -05:00
|
|
|
If(cont.source.stb & (cont.source.charisk == 0b0001),
|
|
|
|
det.eq(cont.source.data)
|
2014-11-11 06:26:32 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
# descrambler
|
2015-01-16 17:52:41 -05:00
|
|
|
self.scrambler = scrambler = LiteSATAScrambler(link_description(32))
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-11-11 06:26:32 -05:00
|
|
|
# check CRC
|
2015-01-16 17:52:41 -05:00
|
|
|
self.crc = crc = LiteSATACRCChecker(link_description(32))
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-05 14:26:09 -05:00
|
|
|
sop = Signal()
|
2014-12-22 14:58:38 -05:00
|
|
|
eop = Signal()
|
2014-12-05 14:26:09 -05:00
|
|
|
self.sync += \
|
2014-12-22 14:58:38 -05:00
|
|
|
If(fsm.ongoing("IDLE"),
|
|
|
|
sop.eq(1),
|
|
|
|
).Elif(fsm.ongoing("COPY"),
|
|
|
|
If(scrambler.sink.stb & scrambler.sink.ack,
|
|
|
|
sop.eq(0)
|
|
|
|
)
|
2014-12-05 14:26:09 -05:00
|
|
|
)
|
2014-12-22 14:58:38 -05:00
|
|
|
self.comb += eop.eq(det == primitives["EOF"])
|
2014-12-05 14:26:09 -05:00
|
|
|
|
2014-12-25 07:11:22 -05:00
|
|
|
crc_error = Signal()
|
|
|
|
self.sync += \
|
|
|
|
If(crc.source.stb & crc.source.eop & crc.source.ack,
|
|
|
|
crc_error.eq(crc.source.error)
|
|
|
|
)
|
|
|
|
|
2014-12-05 15:27:26 -05:00
|
|
|
# small fifo to manage HOLD
|
2014-12-18 19:35:18 -05:00
|
|
|
self.fifo = SyncFIFO(link_description(32), 32)
|
2014-12-05 15:27:26 -05:00
|
|
|
|
2014-11-11 06:26:32 -05:00
|
|
|
# graph
|
2014-12-05 14:26:09 -05:00
|
|
|
self.comb += [
|
2014-12-03 09:29:01 -05:00
|
|
|
cont.source.ack.eq(1),
|
|
|
|
Record.connect(scrambler.source, crc.sink),
|
2014-12-05 15:27:26 -05:00
|
|
|
Record.connect(crc.source, self.fifo.sink),
|
|
|
|
Record.connect(self.fifo.source, self.source)
|
2014-11-11 06:26:32 -05:00
|
|
|
]
|
2014-12-22 14:58:38 -05:00
|
|
|
cont_source_data_d = Signal(32)
|
|
|
|
self.sync += \
|
2014-12-25 06:28:06 -05:00
|
|
|
If(cont.source.stb & (det == 0),
|
2014-12-22 14:58:38 -05:00
|
|
|
scrambler.sink.d.eq(cont.source.data)
|
|
|
|
)
|
2014-11-04 11:35:46 -05:00
|
|
|
|
2014-12-03 09:29:01 -05:00
|
|
|
# FSM
|
2014-11-04 11:35:46 -05:00
|
|
|
fsm.act("IDLE",
|
2014-12-05 14:26:09 -05:00
|
|
|
scrambler.reset.eq(1),
|
2014-12-03 09:29:01 -05:00
|
|
|
If(det == primitives["X_RDY"],
|
|
|
|
NextState("RDY")
|
2014-11-11 06:26:32 -05:00
|
|
|
)
|
2014-11-04 11:35:46 -05:00
|
|
|
)
|
2014-12-03 09:29:01 -05:00
|
|
|
fsm.act("RDY",
|
|
|
|
insert.eq(primitives["R_RDY"]),
|
|
|
|
If(det == primitives["SOF"],
|
2014-12-22 14:58:38 -05:00
|
|
|
NextState("WAIT_FIRST")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("WAIT_FIRST",
|
|
|
|
insert.eq(primitives["R_IP"]),
|
2014-12-25 06:28:06 -05:00
|
|
|
If(cont.source.stb & (det == 0),
|
2014-12-03 09:29:01 -05:00
|
|
|
NextState("COPY")
|
2014-11-11 12:47:34 -05:00
|
|
|
)
|
2014-11-04 11:35:46 -05:00
|
|
|
)
|
2015-01-14 03:19:41 -05:00
|
|
|
self.comb += [
|
|
|
|
scrambler.sink.sop.eq(sop),
|
|
|
|
scrambler.sink.eop.eq(eop)
|
|
|
|
]
|
2014-12-03 09:29:01 -05:00
|
|
|
fsm.act("COPY",
|
2014-12-25 06:28:06 -05:00
|
|
|
scrambler.sink.stb.eq(cont.source.stb & ((det == 0) | eop)),
|
2014-12-05 14:26:09 -05:00
|
|
|
insert.eq(primitives["R_IP"]),
|
2014-12-03 09:29:01 -05:00
|
|
|
If(det == primitives["HOLD"],
|
|
|
|
insert.eq(primitives["HOLDA"])
|
|
|
|
).Elif(det == primitives["EOF"],
|
|
|
|
NextState("WTRM")
|
2014-12-05 15:27:26 -05:00
|
|
|
).Elif(self.fifo.fifo.level > 8,
|
|
|
|
insert.eq(primitives["HOLD"])
|
2014-11-11 06:26:32 -05:00
|
|
|
)
|
2014-11-04 11:35:46 -05:00
|
|
|
)
|
2014-12-03 09:29:01 -05:00
|
|
|
fsm.act("EOF",
|
2014-12-25 07:11:22 -05:00
|
|
|
insert.eq(primitives["R_IP"]),
|
2014-12-03 09:29:01 -05:00
|
|
|
If(det == primitives["WTRM"],
|
|
|
|
NextState("WTRM")
|
2014-11-11 06:26:32 -05:00
|
|
|
)
|
2014-11-04 11:35:46 -05:00
|
|
|
)
|
2014-12-03 09:29:01 -05:00
|
|
|
fsm.act("WTRM",
|
2014-12-25 07:11:22 -05:00
|
|
|
insert.eq(primitives["R_IP"]),
|
|
|
|
If(~crc_error,
|
|
|
|
NextState("R_OK")
|
|
|
|
).Else(
|
|
|
|
NextState("R_ERR")
|
|
|
|
)
|
|
|
|
)
|
|
|
|
fsm.act("R_OK",
|
2014-12-03 09:29:01 -05:00
|
|
|
insert.eq(primitives["R_OK"]),
|
|
|
|
If(det == primitives["SYNC"],
|
2014-11-04 11:35:46 -05:00
|
|
|
NextState("IDLE")
|
2014-11-11 06:26:32 -05:00
|
|
|
)
|
|
|
|
)
|
2014-12-25 07:11:22 -05:00
|
|
|
fsm.act("R_ERR",
|
|
|
|
insert.eq(primitives["R_ERR"]),
|
|
|
|
If(det == primitives["SYNC"],
|
|
|
|
NextState("IDLE")
|
|
|
|
)
|
|
|
|
)
|
2014-11-11 06:26:32 -05:00
|
|
|
|
2014-12-03 09:29:01 -05:00
|
|
|
# to TX
|
|
|
|
self.comb += [
|
|
|
|
self.to_tx.idle.eq(fsm.ongoing("IDLE")),
|
|
|
|
self.to_tx.insert.eq(insert),
|
|
|
|
self.to_tx.det.eq(det)
|
|
|
|
]
|
|
|
|
|
2015-01-16 17:52:41 -05:00
|
|
|
class LiteSATALink(Module):
|
2014-12-23 12:26:07 -05:00
|
|
|
def __init__(self, phy):
|
2015-01-16 17:52:41 -05:00
|
|
|
self.tx = LiteSATALinkTX(phy)
|
|
|
|
self.rx = LiteSATALinkRX(phy)
|
2014-12-03 09:29:01 -05:00
|
|
|
self.comb += Record.connect(self.rx.to_tx, self.tx.from_rx)
|
|
|
|
self.sink, self.source = self.tx.sink, self.rx.source
|