2020-09-28 07:12:07 -04:00
|
|
|
#
|
|
|
|
# This file is part of LiteX.
|
|
|
|
#
|
2021-07-14 03:37:00 -04:00
|
|
|
# Copyright (c) 2021 Florent Kermarrec <florent@enjoy-digital.fr>
|
2020-09-28 07:12:07 -04:00
|
|
|
# SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
|
2021-07-14 03:37:00 -04:00
|
|
|
from migen.fhdl.module import Module
|
|
|
|
from migen.genlib.resetsync import AsyncResetSynchronizer
|
|
|
|
|
|
|
|
from litex.build.io import *
|
|
|
|
|
|
|
|
# Gowin AsyncResetSynchronizer ---------------------------------------------------------------------
|
|
|
|
|
|
|
|
class GowinAsyncResetSynchronizerImpl(Module):
|
|
|
|
def __init__(self, cd, async_reset):
|
|
|
|
rst1 = Signal()
|
|
|
|
self.specials += [
|
|
|
|
Instance("DFFP",
|
|
|
|
i_D = 0,
|
|
|
|
i_PRESET = async_reset,
|
|
|
|
i_CLK = cd.clk,
|
2022-10-24 03:33:22 -04:00
|
|
|
o_Q = rst1
|
|
|
|
),
|
2021-07-14 03:37:00 -04:00
|
|
|
Instance("DFFP",
|
|
|
|
i_D = rst1,
|
|
|
|
i_PRESET = async_reset,
|
|
|
|
i_CLK = cd.clk,
|
|
|
|
o_Q = cd.rst)
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
class GowinAsyncResetSynchronizer:
|
|
|
|
@staticmethod
|
|
|
|
def lower(dr):
|
|
|
|
return GowinAsyncResetSynchronizerImpl(dr.cd, dr.async_reset)
|
|
|
|
|
2021-07-14 04:01:54 -04:00
|
|
|
# Gowin DDR Input ----------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
class GowinDDRInputImpl(Module):
|
|
|
|
def __init__(self, i, o1, o2, clk):
|
|
|
|
self.specials += Instance("IDDR",
|
|
|
|
i_CLK = clk,
|
|
|
|
i_D = i,
|
|
|
|
o_Q0 = o1,
|
|
|
|
o_Q1 = o2,
|
|
|
|
)
|
|
|
|
|
|
|
|
class GowinDDRInput:
|
|
|
|
@staticmethod
|
|
|
|
def lower(dr):
|
|
|
|
return GowinInputImpl(dr.i, dr.o1, dr.o2, dr.clk)
|
|
|
|
|
|
|
|
# Gowin DDR Output ---------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
class GowinDDROutputImpl(Module):
|
|
|
|
def __init__(self, i1, i2, o, clk):
|
|
|
|
self.specials += Instance("ODDR",
|
|
|
|
i_CLK = clk,
|
|
|
|
i_D0 = i1,
|
|
|
|
i_D1 = i2,
|
|
|
|
o_Q0 = o,
|
|
|
|
)
|
|
|
|
|
|
|
|
class GowinDDROutput:
|
|
|
|
@staticmethod
|
|
|
|
def lower(dr):
|
|
|
|
return GowinDDROutputImpl(dr.i1, dr.i2, dr.o, dr.clk)
|
|
|
|
|
2021-09-20 08:14:06 -04:00
|
|
|
# Gowin Differential Input -------------------------------------------------------------------------
|
|
|
|
|
|
|
|
class GowinDifferentialInputImpl(Module):
|
|
|
|
def __init__(self, i_p, i_n, o):
|
|
|
|
self.specials += Instance("TLVDS_IBUF",
|
|
|
|
i_I = i_p,
|
|
|
|
i_IB = i_n,
|
|
|
|
o_O = o,
|
|
|
|
)
|
|
|
|
|
|
|
|
class GowinDifferentialInput:
|
|
|
|
@staticmethod
|
|
|
|
def lower(dr):
|
|
|
|
return GowinDifferentialInputImpl(dr.i_p, dr.i_n, dr.o)
|
|
|
|
|
2022-08-05 02:22:17 -04:00
|
|
|
# Gowin Differential Output ------------------------------------------------------------------------
|
2021-09-20 08:14:06 -04:00
|
|
|
|
|
|
|
class GowinDifferentialOutputImpl(Module):
|
|
|
|
def __init__(self, i, o_p, o_n):
|
|
|
|
self.specials += Instance("TLVDS_OBUF",
|
|
|
|
i_I = i,
|
|
|
|
o_O = o_p,
|
|
|
|
o_OB = o_n,
|
|
|
|
)
|
|
|
|
|
|
|
|
class GowinDifferentialOutput:
|
|
|
|
@staticmethod
|
|
|
|
def lower(dr):
|
|
|
|
return GowinDifferentialOutputImpl(dr.i, dr.o_p, dr.o_n)
|
|
|
|
|
2021-07-14 03:37:00 -04:00
|
|
|
# Gowin Special Overrides --------------------------------------------------------------------------
|
|
|
|
|
|
|
|
gowin_special_overrides = {
|
|
|
|
AsyncResetSynchronizer: GowinAsyncResetSynchronizer,
|
2021-07-14 04:01:54 -04:00
|
|
|
DDRInput: GowinDDRInput,
|
|
|
|
DDROutput: GowinDDROutput,
|
2021-09-20 08:14:06 -04:00
|
|
|
DifferentialInput: GowinDifferentialInput,
|
|
|
|
DifferentialOutput: GowinDifferentialOutput,
|
|
|
|
}
|