mirror of
https://github.com/m-labs/artiq.git
synced 2025-01-22 16:46:42 +08:00
transceiver: init eem media
This commit is contained in:
parent
7ad32d903a
commit
98dff30e88
485
artiq/gateware/drtio/transceiver/eem_serdes.py
Normal file
485
artiq/gateware/drtio/transceiver/eem_serdes.py
Normal file
@ -0,0 +1,485 @@
|
||||
from migen import *
|
||||
from migen.genlib.cdc import MultiReg
|
||||
from migen.genlib.io import DifferentialInput, DifferentialOutput
|
||||
from migen.genlib.fifo import AsyncFIFO
|
||||
from misoc.cores import gpio
|
||||
from misoc.interconnect.csr import *
|
||||
from misoc.cores.code_8b10b import SingleEncoder, Decoder
|
||||
from artiq.gateware.drtio.core import TransceiverInterface, ChannelInterface
|
||||
|
||||
|
||||
class RXPhy(Module):
|
||||
def __init__(self, i_pads):
|
||||
self.o = [Signal() for _ in range(4)]
|
||||
|
||||
for i in range(4):
|
||||
self.specials += Instance("IBUFDS",
|
||||
i_I=i_pads[i].p,
|
||||
i_IB=i_pads[i].n,
|
||||
o_O=self.o[i],
|
||||
)
|
||||
|
||||
|
||||
class TXPhy(Module):
|
||||
def __init__(self, o_pads):
|
||||
self.i = [Signal() for _ in range(4)]
|
||||
self.t = [Signal() for _ in range(4)]
|
||||
|
||||
for i in range(4):
|
||||
self.specials += Instance("OBUFTDS",
|
||||
i_I=self.i[i],
|
||||
o_O=o_pads[i].p,
|
||||
o_OB=o_pads[i].n,
|
||||
# Always chain the 3-states input to serializer
|
||||
# Vivado will complain otherwise
|
||||
i_T=self.t[i],
|
||||
)
|
||||
|
||||
|
||||
class RXSerdes(Module):
|
||||
def __init__(self):
|
||||
self.rxdata = [ Signal(10) for _ in range(4) ]
|
||||
self.ser_in_no_dly = [ Signal() for _ in range(4) ]
|
||||
self.ld = [ Signal() for _ in range(4) ]
|
||||
self.cnt_in = [ Signal(5) for _ in range(4) ]
|
||||
self.cnt_out = [ Signal(5) for _ in range(4) ]
|
||||
self.bitslip = [ Signal() for _ in range(4) ]
|
||||
|
||||
ser_in = [ Signal() for _ in range(4) ]
|
||||
shifts = [ Signal(2) for _ in range(4) ]
|
||||
|
||||
for i in range(4):
|
||||
self.specials += [
|
||||
# Master deserializer
|
||||
Instance("ISERDESE2",
|
||||
p_DATA_RATE="DDR",
|
||||
p_DATA_WIDTH=10,
|
||||
p_INTERFACE_TYPE="NETWORKING",
|
||||
p_NUM_CE=1,
|
||||
p_SERDES_MODE="MASTER",
|
||||
p_IOBDELAY="IFD",
|
||||
o_Q1=self.rxdata[i][9],
|
||||
o_Q2=self.rxdata[i][8],
|
||||
o_Q3=self.rxdata[i][7],
|
||||
o_Q4=self.rxdata[i][6],
|
||||
o_Q5=self.rxdata[i][5],
|
||||
o_Q6=self.rxdata[i][4],
|
||||
o_Q7=self.rxdata[i][3],
|
||||
o_Q8=self.rxdata[i][2],
|
||||
o_SHIFTOUT1=shifts[i][0],
|
||||
o_SHIFTOUT2=shifts[i][1],
|
||||
i_DDLY=ser_in[i],
|
||||
i_BITSLIP=self.bitslip[i],
|
||||
i_CLK=ClockSignal("eem_sys5x"),
|
||||
i_CLKB=~ClockSignal("eem_sys5x"),
|
||||
i_CE1=1,
|
||||
i_RST=ResetSignal("eem_sys"),
|
||||
i_CLKDIV=ClockSignal("eem_sys")),
|
||||
|
||||
# Slave deserializer
|
||||
Instance("ISERDESE2",
|
||||
p_DATA_RATE="DDR",
|
||||
p_DATA_WIDTH=10,
|
||||
p_INTERFACE_TYPE="NETWORKING",
|
||||
p_NUM_CE=1,
|
||||
p_SERDES_MODE="SLAVE",
|
||||
p_IOBDELAY="IFD",
|
||||
o_Q3=self.rxdata[i][1],
|
||||
o_Q4=self.rxdata[i][0],
|
||||
i_BITSLIP=self.bitslip[i],
|
||||
i_CLK=ClockSignal("eem_sys5x"),
|
||||
i_CLKB=~ClockSignal("eem_sys5x"),
|
||||
i_CE1=1,
|
||||
i_RST=ResetSignal("eem_sys"),
|
||||
i_CLKDIV=ClockSignal("eem_sys"),
|
||||
i_SHIFTIN1=shifts[i][0],
|
||||
i_SHIFTIN2=shifts[i][1]),
|
||||
|
||||
# Tunable delay
|
||||
Instance("IDELAYE2",
|
||||
p_DELAY_SRC="IDATAIN",
|
||||
p_SIGNAL_PATTERN="DATA",
|
||||
p_CINVCTRL_SEL="FALSE",
|
||||
p_HIGH_PERFORMANCE_MODE="TRUE",
|
||||
# REFCLK refers to the clock source of IDELAYCTRL
|
||||
p_REFCLK_FREQUENCY=200.0,
|
||||
p_PIPE_SEL="FALSE",
|
||||
p_IDELAY_TYPE="VAR_LOAD",
|
||||
p_IDELAY_VALUE=0,
|
||||
|
||||
i_C=ClockSignal(),
|
||||
i_LD=self.ld[i],
|
||||
i_CE=0,
|
||||
i_LDPIPEEN=0,
|
||||
i_INC=1, # Always increment
|
||||
|
||||
# Set the optimal delay tap via the aligner
|
||||
i_CNTVALUEIN=self.cnt_in[i],
|
||||
# Allow the aligner to check the tap value
|
||||
o_CNTVALUEOUT=self.cnt_out[i],
|
||||
|
||||
i_IDATAIN=self.ser_in_no_dly[i],
|
||||
o_DATAOUT=ser_in[i]
|
||||
),
|
||||
|
||||
# IDELAYCTRL is with the clocking
|
||||
]
|
||||
|
||||
|
||||
class TXSerdes(Module):
|
||||
def __init__(self):
|
||||
self.txdata = [ Signal(5) for _ in range(4) ]
|
||||
self.ser_out = [ Signal() for _ in range(4) ]
|
||||
self.t_out = [ Signal() for _ in range(4) ]
|
||||
|
||||
# TX SERDES
|
||||
for i in range(4):
|
||||
self.specials += Instance("OSERDESE2",
|
||||
p_DATA_RATE_OQ="SDR", p_DATA_RATE_TQ="BUF",
|
||||
p_DATA_WIDTH=5, p_TRISTATE_WIDTH=1,
|
||||
p_INIT_OQ=0b00000,
|
||||
o_OQ=self.ser_out[i],
|
||||
o_TQ=self.t_out[i],
|
||||
i_RST=ResetSignal("eem_sys"),
|
||||
i_CLK=ClockSignal("eem_sys5x"),
|
||||
i_CLKDIV=ClockSignal("eem_sys"),
|
||||
i_D1=self.txdata[i][0],
|
||||
i_D2=self.txdata[i][1],
|
||||
i_D3=self.txdata[i][2],
|
||||
i_D4=self.txdata[i][3],
|
||||
i_D5=self.txdata[i][4],
|
||||
i_TCE=1, i_OCE=1,
|
||||
i_T1=0)
|
||||
|
||||
|
||||
class MultiEncoder(Module):
|
||||
def __init__(self):
|
||||
WORDS = 2
|
||||
# Keep the link layer interface identical to standard encoders
|
||||
self.d = [Signal(8) for _ in range(WORDS)]
|
||||
self.k = [Signal() for _ in range(WORDS)]
|
||||
|
||||
# Alignment control: Keep sending K.28.5
|
||||
self.align = Signal()
|
||||
|
||||
# Output interface is simplified because we have custom physical layer
|
||||
self.output = [Signal(10) for _ in range(WORDS)]
|
||||
|
||||
# Phase of the encoder
|
||||
# Alternate crossbar between encoder and SERDES every cycle
|
||||
self.phase = Signal()
|
||||
|
||||
# Intermediate registers for output and disparity
|
||||
# More significant bits are buffered due to channel geometry
|
||||
# Disparity bit is delayed. The same encoder is shared by 2 SERDES
|
||||
output_bufs = [Signal(5) for _ in range(WORDS)]
|
||||
disp_bufs = [Signal() for _ in range(WORDS)]
|
||||
|
||||
encoders = [SingleEncoder() for _ in range(WORDS)]
|
||||
self.submodules += encoders
|
||||
|
||||
for d, k, output, output_buf, disp_buf, encoder in \
|
||||
zip(self.d, self.k, self.output, output_bufs, disp_bufs, encoders):
|
||||
self.comb += [
|
||||
If(self.align,
|
||||
encoder.d.eq(0xBC),
|
||||
encoder.k.eq(1),
|
||||
).Else(
|
||||
encoder.d.eq(d),
|
||||
encoder.k.eq(k),
|
||||
),
|
||||
|
||||
# Implementing switching crossbar
|
||||
If(self.phase,
|
||||
output.eq(Cat(encoder.output[0:5], output_buf))
|
||||
).Else(
|
||||
output.eq(Cat(output_buf, encoder.output[0:5]))
|
||||
),
|
||||
]
|
||||
# Handle intermediate registers
|
||||
self.sync.eem_sys += [
|
||||
disp_buf.eq(encoder.disp_out),
|
||||
encoder.disp_in.eq(disp_buf),
|
||||
output_buf.eq(encoder.output[5:10]),
|
||||
]
|
||||
|
||||
|
||||
# Unlike the usual 8b10b decoder, it needs to know which SERDES to decode
|
||||
class CrossbarDecoder(Module):
|
||||
def __init__(self):
|
||||
self.raw_input = [ Signal(5) for _ in range(2) ]
|
||||
self.d = Signal(8)
|
||||
self.k = Signal()
|
||||
|
||||
# Notifier signal when group alignmnet is completed
|
||||
self.delay = Signal(2)
|
||||
self.phase = Signal()
|
||||
|
||||
# Optional extra stage for both lanes
|
||||
self.delay_buf = [ Signal(5) for _ in range(2) ]
|
||||
self.sync.eem_sys += [
|
||||
self.delay_buf[idx].eq(self.raw_input[idx]) for idx in range(2)
|
||||
]
|
||||
|
||||
# Intermediate register for input
|
||||
buffer = Signal(5)
|
||||
|
||||
self.submodules.decoder = Decoder()
|
||||
|
||||
# Update phase & synchronous elements
|
||||
self.sync.eem_sys += [
|
||||
If(~self.phase,
|
||||
If(~self.delay[0],
|
||||
buffer.eq(self.raw_input[0]),
|
||||
).Else(
|
||||
buffer.eq(self.delay_buf[0]),
|
||||
)
|
||||
).Else(
|
||||
If(~self.delay[1],
|
||||
buffer.eq(self.raw_input[1]),
|
||||
).Else(
|
||||
buffer.eq(self.delay_buf[1]),
|
||||
)
|
||||
)
|
||||
]
|
||||
|
||||
# Send appropriate input to decoder
|
||||
self.comb += [
|
||||
If(self.phase,
|
||||
If(~self.delay[0],
|
||||
self.decoder.input.eq(Cat(buffer, self.raw_input[0])),
|
||||
).Else(
|
||||
self.decoder.input.eq(Cat(buffer, self.delay_buf[0])),
|
||||
)
|
||||
).Else(
|
||||
If(~self.delay[1],
|
||||
self.decoder.input.eq(Cat(buffer, self.raw_input[1])),
|
||||
).Else(
|
||||
self.decoder.input.eq(Cat(buffer, self.delay_buf[1])),
|
||||
)
|
||||
),
|
||||
]
|
||||
|
||||
self.comb += [
|
||||
self.d.eq(self.decoder.d),
|
||||
self.k.eq(self.decoder.k),
|
||||
]
|
||||
|
||||
|
||||
class SerdesSingle(Module, AutoCSR):
|
||||
def __init__(self, i_pads, o_pads, debug=False):
|
||||
# Modules for the IOB
|
||||
self.submodules.rx_phy = RXPhy(i_pads)
|
||||
self.submodules.tx_phy = TXPhy(o_pads)
|
||||
|
||||
# Serdes Module
|
||||
self.submodules.rx_serdes = RXSerdes()
|
||||
self.submodules.tx_serdes = TXSerdes()
|
||||
|
||||
# CSR for delay & bitslip
|
||||
self.bitslip_sel = CSRStorage(4)
|
||||
self.bitslip = CSR()
|
||||
|
||||
for i in range(4):
|
||||
self.specials += MultiReg(
|
||||
self.bitslip_sel.storage[i] & self.bitslip.re,
|
||||
self.rx_serdes.bitslip[i], "eem_sys")
|
||||
|
||||
self.dly_cnt_in_sel = CSRStorage(4)
|
||||
self.dly_cnt_in = CSRStorage(5)
|
||||
self.dly_ld = CSR()
|
||||
|
||||
for i in range(4):
|
||||
self.comb += [
|
||||
self.rx_serdes.cnt_in[i].eq(self.dly_cnt_in.storage),
|
||||
self.rx_serdes.ld[i].eq(self.dly_cnt_in_sel.storage[i] & self.dly_ld.re),
|
||||
]
|
||||
|
||||
self.dly_cnt_out_sel = CSRStorage(4)
|
||||
self.dly_cnt_out = CSRStatus(5)
|
||||
|
||||
self.comb += Case(self.dly_cnt_out_sel.storage, {
|
||||
(1 << idx): self.dly_cnt_out.status.eq(self.rx_serdes.cnt_out[idx]) for idx in range(4)
|
||||
})
|
||||
|
||||
# CSR for global decoding phase
|
||||
# This is to determine if this cycle should decode SERDES 0 or 1
|
||||
self.decoder_dly = CSRStorage(4)
|
||||
dec_dly_cdc = Signal(4)
|
||||
self.specials += MultiReg(self.decoder_dly.storage, dec_dly_cdc, "eem_sys")
|
||||
|
||||
# Encoder/Decodfer interfaces
|
||||
self.submodules.encoder = MultiEncoder()
|
||||
self.submodules.decoders = decoders = Array(CrossbarDecoder() for _ in range(2))
|
||||
|
||||
# Wire up the IOB to serdes modules
|
||||
for i in range(4):
|
||||
self.comb += [
|
||||
self.tx_phy.i[i].eq(self.tx_serdes.ser_out[i]),
|
||||
self.tx_phy.t[i].eq(self.tx_serdes.t_out[i]),
|
||||
|
||||
self.rx_serdes.ser_in_no_dly[i].eq(self.rx_phy.o[i]),
|
||||
]
|
||||
|
||||
# Control decoders phase
|
||||
self.comb += [
|
||||
decoders[0].delay.eq(dec_dly_cdc[:2]),
|
||||
decoders[1].delay.eq(dec_dly_cdc[2:]),
|
||||
]
|
||||
|
||||
# Route encoded symbols to TXSerdes
|
||||
self.comb += [
|
||||
self.tx_serdes.txdata[0].eq(self.encoder.output[0][:5]),
|
||||
self.tx_serdes.txdata[1].eq(self.encoder.output[0][5:]),
|
||||
self.tx_serdes.txdata[2].eq(self.encoder.output[1][:5]),
|
||||
self.tx_serdes.txdata[3].eq(self.encoder.output[1][5:]),
|
||||
]
|
||||
|
||||
# Truncate RX, controllable via CSR
|
||||
self.select_odd = CSRStorage(4)
|
||||
select_odd_cdc = Signal(4)
|
||||
self.specials += MultiReg(self.select_odd.storage, select_odd_cdc, "eem_sys")
|
||||
|
||||
decimated_rxdata = [ Signal(5) for _ in range(4) ]
|
||||
for i in range(4):
|
||||
self.comb += decimated_rxdata[i].eq(Mux(select_odd_cdc[i],
|
||||
self.rx_serdes.rxdata[i][1::2], self.rx_serdes.rxdata[i][0::2]))
|
||||
|
||||
# Route RXSerdes to decoder
|
||||
self.comb += [
|
||||
decoders[i//2].raw_input[i%2].eq(decimated_rxdata[i]) for i in range(4)
|
||||
]
|
||||
|
||||
# Always send out K.28.5 if not aligned
|
||||
self.send_align = CSRStorage(reset=1)
|
||||
self.specials += MultiReg(self.send_align.storage, self.encoder.align, "eem_sys")
|
||||
|
||||
# Alternate phase
|
||||
phase = Signal()
|
||||
# Assign to encoder and decoder
|
||||
self.comb += [
|
||||
self.encoder.phase.eq(phase),
|
||||
self.decoders[0].phase.eq(phase),
|
||||
self.decoders[1].phase.eq(phase),
|
||||
]
|
||||
|
||||
# Phase increment & reset mechanism
|
||||
# Reset to 0 when externally triggered
|
||||
self.phase_rst = Signal()
|
||||
self.sync.eem_sys += If(self.phase_rst,
|
||||
phase.eq(0),
|
||||
).Else(
|
||||
phase.eq(~phase)
|
||||
)
|
||||
self.phase_out = Signal()
|
||||
self.comb += self.phase_out.eq(phase)
|
||||
|
||||
# Interleave data/ctrl update
|
||||
self.read_word = CSRStorage(2)
|
||||
self.aligned = CSRStatus()
|
||||
|
||||
rx_d = Signal(8)
|
||||
rx_k = Signal()
|
||||
|
||||
rx_d_prev = Signal(8)
|
||||
rx_k_prev = Signal()
|
||||
|
||||
read_word_cdc = Signal(2)
|
||||
self.specials += MultiReg(self.read_word.storage, read_word_cdc)
|
||||
|
||||
self.sync.eem_sys += [
|
||||
If(~phase ^ read_word_cdc[0],
|
||||
rx_d.eq(decoders[read_word_cdc[1]].d),
|
||||
rx_k.eq(decoders[read_word_cdc[1]].k),
|
||||
rx_d_prev.eq(rx_d),
|
||||
rx_k_prev.eq(rx_k),
|
||||
)
|
||||
]
|
||||
|
||||
found_align_symbol = Signal()
|
||||
self.comb += found_align_symbol.eq(
|
||||
(rx_d == 0xBC) & (rx_d_prev == 0xBC)
|
||||
& (rx_k == 1) & (rx_k_prev == 1))
|
||||
|
||||
self.specials += MultiReg(found_align_symbol, self.aligned.status)
|
||||
|
||||
|
||||
layout = [
|
||||
("sat_clk_rdy", 2, "satellite"),
|
||||
("phase_rdy", 3, "master"),
|
||||
("sat_rst", 4, "master"),
|
||||
("mst_clk_rdy", 5, "master"),
|
||||
("align_mst", 6, "satellite"),
|
||||
("align_sat", 7, "master"),
|
||||
]
|
||||
|
||||
class EEMSerdes(Module, TransceiverInterface):
|
||||
def __init__(self, platform, eem, eem_aux, role="master", start_idx=0):
|
||||
self.rx_ready = CSRStorage()
|
||||
|
||||
# Request resources
|
||||
# TODO: Expand to support multiple EFCs
|
||||
i_pads = [
|
||||
platform.request("eem{}_fmc_data_in".format(eem), i) for i in range(4)
|
||||
]
|
||||
o_pads = [
|
||||
platform.request("eem{}_fmc_data_out".format(eem), i) for i in range(4)
|
||||
]
|
||||
|
||||
self.submodules.serdes = SerdesSingle(i_pads, o_pads)
|
||||
self.submodules.aux = EEMAux(platform, eem_aux, role=role)
|
||||
|
||||
# TODO: Move global phase here
|
||||
if role == "master":
|
||||
self.comb += self.aux.phase_in.eq(self.serdes.phase_out)
|
||||
elif role == "satellite":
|
||||
self.comb += self.serdes.phase_rst.eq(self.aux.phase_rst)
|
||||
|
||||
chan_if = ChannelInterface(self.serdes.encoder, self.serdes.decoders)
|
||||
self.comb += chan_if.rx_ready.eq(self.rx_ready.storage)
|
||||
channel_interfaces = [chan_if]
|
||||
|
||||
TransceiverInterface.__init__(self, channel_interfaces, start_idx=start_idx)
|
||||
|
||||
self.comb += [
|
||||
getattr(self, "cd_rtio_rx" + str(start_idx)).clk.eq(ClockSignal("eem_sys")),
|
||||
getattr(self, "cd_rtio_rx" + str(start_idx)).rst.eq(ResetSignal("eem_sys"))
|
||||
]
|
||||
|
||||
|
||||
class EEMAux(Module, AutoCSR):
|
||||
def __init__(self, platform, eem_aux, role="master"):
|
||||
for name, _, src in layout:
|
||||
if name != "sat_rst":
|
||||
tmp = Signal()
|
||||
pad = platform.request(("eem{}_fmc_"+name).format(eem_aux))
|
||||
if src == role:
|
||||
self.specials += DifferentialOutput(tmp, pad.p, pad.n)
|
||||
setattr(self.submodules, name, gpio.GPIOOut(tmp))
|
||||
else:
|
||||
self.specials += DifferentialInput(pad.p, pad.n, tmp)
|
||||
setattr(self.submodules, name, gpio.GPIOIn(tmp))
|
||||
|
||||
sat_rst_pad = platform.request("eem{}_fmc_sat_rst".format(eem_aux))
|
||||
sat_rst_tmp = Signal()
|
||||
if role == "master":
|
||||
self.phase_in = Signal()
|
||||
self.sat_phase_rst = CSR()
|
||||
sat_phase_rst_r = Signal()
|
||||
sat_phase_rst_rr = Signal()
|
||||
sat_phase_rst_cdc = Signal()
|
||||
|
||||
self.specials += MultiReg(self.sat_phase_rst.re, sat_phase_rst_r, "eem_sys")
|
||||
self.sync.eem_sys += sat_phase_rst_rr.eq(sat_phase_rst_r)
|
||||
self.comb += sat_phase_rst_cdc.eq(sat_phase_rst_rr | sat_phase_rst_r)
|
||||
gated_pulse = Signal()
|
||||
self.sync.eem_sys += gated_pulse.eq(sat_phase_rst_cdc & self.phase_in)
|
||||
self.specials += DifferentialOutput(gated_pulse, sat_rst_pad.p, sat_rst_pad.n)
|
||||
|
||||
elif role == "satellite":
|
||||
self.phase_rst = Signal()
|
||||
phase_in_raw = Signal()
|
||||
self.specials += DifferentialInput(sat_rst_pad.p, sat_rst_pad.n, phase_in_raw)
|
||||
self.specials += MultiReg(phase_in_raw, self.phase_rst, "eem_sys")
|
||||
else:
|
||||
ValueError("Invalid role type")
|
Loading…
Reference in New Issue
Block a user