From 2612fd1e721eda17a9e176f8d47b5ce3d1a1110d Mon Sep 17 00:00:00 2001 From: Sebastien Bourdeauducq Date: Mon, 28 May 2018 22:42:27 +0800 Subject: [PATCH] rtio: add grabber deserializer and WIP PHY encapsulation --- artiq/gateware/grabber/__init__.py | 0 .../gateware/grabber/deserializer_7series.py | 124 ++++++++++++++++++ artiq/gateware/rtio/phy/grabber.py | 16 +++ 3 files changed, 140 insertions(+) create mode 100644 artiq/gateware/grabber/__init__.py create mode 100644 artiq/gateware/grabber/deserializer_7series.py create mode 100644 artiq/gateware/rtio/phy/grabber.py diff --git a/artiq/gateware/grabber/__init__.py b/artiq/gateware/grabber/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/artiq/gateware/grabber/deserializer_7series.py b/artiq/gateware/grabber/deserializer_7series.py new file mode 100644 index 000000000..b7336a578 --- /dev/null +++ b/artiq/gateware/grabber/deserializer_7series.py @@ -0,0 +1,124 @@ +from migen import * +from migen.genlib.cdc import MultiReg +from migen.genlib.resetsync import AsyncResetSynchronizer + +from misoc.interconnect.csr import * + + +# See: +# http://www.volkerschatz.com/hardware/clink.html + +class Deserializer(Module, AutoCSR): + def __init__(self, pins): + self.pll_reset = CSRStorage(reset=1) + self.pll_locked = CSRStatus() + self.phase_shift = CSR() + self.phase_shift_done = CSRStatus(reset=1) + self.clk_sampled = CSRStatus(7) + + self.q_clk = Signal(7) + self.q = Signal(7*len(pins.sdi_p)) + + self.clock_domains.cd_cl = ClockDomain() + self.clock_domains.cd_cl7x = ClockDomain() + + # # # + + clk_se = Signal() + self.specials += Instance("IBUFDS", + i_I=pins.clk_p, i_IB=pins.clk_n, o_O=clk_se) + + clk_se_iserdes = Signal() + self.specials += [ + Instance("ISERDESE2", + p_DATA_WIDTH=7, p_DATA_RATE="SDR", + p_SERDES_MODE="MASTER", p_INTERFACE_TYPE="NETWORKING", + p_NUM_CE=1, + + i_D=clk_se, + o_O=clk_se_iserdes, + i_CE1=1, + i_CLKDIV=ClockSignal("cl"), i_RST=ResetSignal("cl"), + i_CLK=ClockSignal("cl7x"), i_CLKB=~ClockSignal("cl7x"), + o_Q1=self.q_clk[6], + o_Q2=self.q_clk[5], o_Q3=self.q_clk[4], + o_Q4=self.q_clk[3], o_Q5=self.q_clk[2], + o_Q6=self.q_clk[1], o_Q7=self.q_clk[0] + ) + ] + + sdi_se = Signal(len(pins.sdi_p)) + for i in range(len(pins.sdi_p)): + self.specials += [ + Instance("IBUFDS", i_I=pins.sdi_p[i], i_IB=pins.sdi_n[i], + o_O=sdi_se[i]), + Instance("ISERDESE2", + p_DATA_WIDTH=7, p_DATA_RATE="SDR", + p_SERDES_MODE="MASTER", p_INTERFACE_TYPE="NETWORKING", + p_NUM_CE=1, + + i_D=sdi_se[i], + i_CE1=1, + i_CLKDIV=ClockSignal("cl"), i_RST=ResetSignal("cl"), + i_CLK=ClockSignal("cl7x"), i_CLKB=~ClockSignal("cl7x"), + o_Q1=self.q[7*i+6], + o_Q2=self.q[7*i+5], o_Q3=self.q[7*i+4], + o_Q4=self.q[7*i+3], o_Q5=self.q[7*i+2], + o_Q6=self.q[7*i+1], o_Q7=self.q[7*i+0] + ) + ] + + # CL clock frequency 40-85MHz + # A7-2 MMCM VCO frequency 600-1440MHz + # A7-2 PLL VCO frequency 800-1866MHz + # with current MMCM settings, CL frequency limited to 40-~68MHz + # TODO: switch to the PLL, whose VCO range better matches the CL + # clock frequencies. Needs DRP for dynamic phase shift, see XAPP888. + pll_reset = Signal(reset=1) + mmcm_fb = Signal() + mmcm_locked = Signal() + mmcm_ps_psdone = Signal() + cl_clk = Signal() + cl7x_clk = Signal() + phase = 257.0 + self.specials += [ + Instance("MMCME2_ADV", + p_CLKIN1_PERIOD=18.0, + i_CLKIN1=clk_se_iserdes, + i_RST=pll_reset, + i_CLKINSEL=1, # yes, 1=CLKIN1 0=CLKIN2 + + p_CLKFBOUT_MULT_F=21.0, + p_DIVCLK_DIVIDE=1, + o_LOCKED=mmcm_locked, + + o_CLKFBOUT=mmcm_fb, i_CLKFBIN=mmcm_fb, + + p_CLKOUT0_USE_FINE_PS="TRUE", + p_CLKOUT0_DIVIDE_F=21.0, + p_CLKOUT0_PHASE=phase, + o_CLKOUT0=cl_clk, + + p_CLKOUT1_USE_FINE_PS="TRUE", + p_CLKOUT1_DIVIDE=3, + p_CLKOUT1_PHASE=phase*7 % 360.0, + o_CLKOUT1=cl7x_clk, + + i_PSCLK=ClockSignal(), + i_PSEN=self.phase_shift.re, + i_PSINCDEC=self.phase_shift.r, + o_PSDONE=mmcm_ps_psdone, + ), + Instance("BUFG", i_I=cl_clk, o_O=self.cd_cl.clk), + Instance("BUFG", i_I=cl7x_clk, o_O=self.cd_cl7x.clk), + AsyncResetSynchronizer(self.cd_cl, ~mmcm_locked), + ] + self.sync += [ + If(self.phase_shift.re, self.phase_shift_done.status.eq(0)), + If(mmcm_ps_psdone, self.phase_shift_done.status.eq(1)) + ] + self.specials += MultiReg(self.q_clk, self.clk_sampled.status) + + self.specials += MultiReg(mmcm_locked, self.pll_locked.status) + pll_reset.attr.add("no_retiming") + self.sync += pll_reset.eq(self.pll_reset.storage) diff --git a/artiq/gateware/rtio/phy/grabber.py b/artiq/gateware/rtio/phy/grabber.py new file mode 100644 index 000000000..9d1a151d7 --- /dev/null +++ b/artiq/gateware/rtio/phy/grabber.py @@ -0,0 +1,16 @@ +from migen import * + +from artiq.gateware.rtio import rtlink +from artiq.gateware.grabber import deserializer_7series + + +class Grabber(Module): + def __init__(self, pins): + self.config = rtlink.Interface(rtlink.OInterface(10)) + self.gate_data = rtlink.Interface(rtlink.OInterface(1), + rtlink.IInterface(10)) + + self.submodules.deserializer = deserializer_7series.Deserializer(pins) + + def get_csrs(self): + return self.deserializer.get_csrs()