Add vcu128 target (#497)

Add initial VCU128 support.
This commit is contained in:
Chen 2023-05-26 04:25:44 +08:00 committed by GitHub
parent 9362e589ac
commit 2ae2dfa6a3
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 289 additions and 0 deletions

View file

@ -0,0 +1,147 @@
#
# This file is part of LiteX-Boards.
#
# Copyright (c) 2020 Fei Gao <feig@princeton.edu>
# Copyright (c) 2020 Florent Kermarrec <florent@enjoy-digital.fr>
# Copyright (c) 2022 Jiajie Chen <c@jia.je>
# SPDX-License-Identifier: BSD-2-Clause
from litex.build.generic_platform import *
from litex.build.xilinx import XilinxUSPPlatform, VivadoProgrammer
# IOs ----------------------------------------------------------------------------------------------
_io = [
# Clk / Rst
("clk100_ddr4", 0,
Subsignal("p", Pins("BH51"), IOStandard("LVDS")),
Subsignal("n", Pins("BJ51"), IOStandard("LVDS")),
),
("clk100_qdr4", 0,
Subsignal("p", Pins("BJ4"), IOStandard("LVDS")),
Subsignal("n", Pins("BK3"), IOStandard("LVDS")),
),
("clk100_rld3", 0,
Subsignal("p", Pins("F35"), IOStandard("LVDS")),
Subsignal("n", Pins("F36"), IOStandard("LVDS")),
),
("cpu_reset", 0, Pins("BM29"), IOStandard("LVCMOS12")),
# Leds
("user_led", 0, Pins("BH24"), IOStandard("LVCMOS18")),
("user_led", 1, Pins("BG24"), IOStandard("LVCMOS18")),
("user_led", 2, Pins("BG25"), IOStandard("LVCMOS18")),
("user_led", 3, Pins("BF25"), IOStandard("LVCMOS18")),
("user_led", 4, Pins("BF26"), IOStandard("LVCMOS18")),
("user_led", 5, Pins("BF27"), IOStandard("LVCMOS18")),
("user_led", 6, Pins("BG27"), IOStandard("LVCMOS18")),
("user_led", 7, Pins("BG28"), IOStandard("LVCMOS18")),
# Serial
("serial", 0,
Subsignal("rx", Pins("BP26"), IOStandard("LVCMOS18")),
Subsignal("rts", Pins("BP22"), IOStandard("LVCMOS18")),
Subsignal("tx", Pins("BN26"), IOStandard("LVCMOS18")),
Subsignal("cts", Pins("BP23"), IOStandard("LVCMOS18")),
),
("serial", 1,
Subsignal("rx", Pins("BK28"), IOStandard("LVCMOS18")),
Subsignal("rts", Pins("BL26"), IOStandard("LVCMOS18")),
Subsignal("tx", Pins("BJ28"), IOStandard("LVCMOS18")),
Subsignal("cts", Pins("BL27"), IOStandard("LVCMOS18")),
),
# DDR4 memory
("ddram", 0,
Subsignal("a", Pins(
"BF50 BD51 BG48 BE50 BE49 BE51 BF53 BG50",
"BF51 BG47 BF47 BG49 BF48 BF52"),
IOStandard("SSTL12_DCI")),
Subsignal("ba", Pins("BE54 BE53"), IOStandard("SSTL12_DCI")),
Subsignal("bg", Pins("BG54"), IOStandard("SSTL12_DCI")),
Subsignal("ras_n", Pins("BJ54"), IOStandard("SSTL12_DCI")),
Subsignal("cas_n", Pins("BH54"), IOStandard("SSTL12_DCI")),
Subsignal("we_n", Pins("BG53"), IOStandard("SSTL12_DCI")),
Subsignal("cs_n", Pins("BP49 BK48"), IOStandard("SSTL12_DCI")), # Clam-shell topology
Subsignal("act_n", Pins("BG52"), IOStandard("SSTL12_DCI")),
#Subsignal("ten", Pins("BJ53"), IOStandard("SSTL12_DCI")),
#Subsignal("alert_n", Pins("BJ52"), IOStandard("SSTL12_DCI")),
#Subsignal("par", Pins("BL48"), IOStandard("SSTL12_DCI")),
Subsignal("dm", Pins(
"BN42 BL47 BH42 BD41 BM28 BM34 BH32 BG29"),
IOStandard("POD12_DCI")),
Subsignal("dq", Pins(
"BM45 BP44 BP47 BN45 BM44 BN44 BN47 BP43",
"BL45 BK44 BL46 BK43 BL43 BJ44 BL42 BJ43",
"BK41 BG44 BG42 BH44 BH45 BG45 BG43 BJ41",
"BE43 BF42 BC42 BF43 BD42 BF45 BE44 BF46",
"BP32 BP29 BP31 BP28 BN32 BM30 BN31 BL30",
"BL32 BP34 BN34 BK33 BL31 BL33 BM33 BK31",
"BJ34 BG35 BH34 BH35 BJ33 BF35 BG34 BF36",
"BF31 BH30 BJ31 BG32 BH31 BF32 BH29 BF33"),
IOStandard("POD12_DCI"),
Misc("PRE_EMPHASIS=RDRV_240"),
Misc("EQUALIZATION=EQ_LEVEL2")),
Subsignal("dqs_p", Pins("BN46 BK45 BH46 BE45 BN29 BL35 BK34 BJ29"),
IOStandard("DIFF_POD12"),
Misc("PRE_EMPHASIS=RDRV_240"),
Misc("EQUALIZATION=EQ_LEVEL2")),
Subsignal("dqs_n", Pins("BP46 BK46 BJ46 BE46 BN30 BM35 BK35 BK30"),
IOStandard("DIFF_POD12"),
Misc("PRE_EMPHASIS=RDRV_240"),
Misc("EQUALIZATION=EQ_LEVEL2")),
Subsignal("clk_p", Pins("BK53"), IOStandard("DIFF_SSTL12_DCI")),
Subsignal("clk_n", Pins("BK54"), IOStandard("DIFF_SSTL12_DCI")),
Subsignal("cke", Pins("BH52"), IOStandard("SSTL12_DCI")),
Subsignal("odt", Pins("BH49"), IOStandard("SSTL12_DCI")),
Subsignal("reset_n", Pins("BH50"), IOStandard("LVCMOS12")),
Misc("SLEW=FAST"),
),
# SGMII Clock
("eth_clocks", 0,
Subsignal("p", Pins("BH27"), IOStandard("LVDS")),
Subsignal("n", Pins("BJ27"), IOStandard("LVDS")),
),
# SGMII Ethernet
("eth", 0,
Subsignal("int_n", Pins("BF22"), IOStandard("LVCMOS18")),
Subsignal("mdio", Pins("BG23"), IOStandard("LVCMOS18")),
Subsignal("mdc", Pins("BN27"), IOStandard("LVCMOS18")),
Subsignal("rx_p", Pins("BJ22"), IOStandard("LVDS")),
Subsignal("rx_n", Pins("BK21"), IOStandard("LVDS")),
Subsignal("tx_p", Pins("BG22"), IOStandard("LVDS")),
Subsignal("tx_n", Pins("BH22"), IOStandard("LVDS")),
),
]
# Connectors ---------------------------------------------------------------------------------------
_connectors = []
# Platform -----------------------------------------------------------------------------------------
class Platform(XilinxUSPPlatform):
default_clk_name = "clk100_ddr4"
default_clk_period = 1e9/100e6
def __init__(self, toolchain="vivado"):
XilinxUSPPlatform.__init__(self, "xcvu37p-fsvh2892-2L-e", _io, _connectors, toolchain="vivado")
def create_programmer(self):
return VivadoProgrammer()
def do_finalize(self, fragment):
XilinxUSPPlatform.do_finalize(self, fragment)
self.add_period_constraint(self.lookup_request("clk100_ddr4", loose=True), 1e9/100e6)
self.add_period_constraint(self.lookup_request("clk100_qdr4", loose=True), 1e9/100e6)
self.add_period_constraint(self.lookup_request("clk100_rld3", loose=True), 1e9/100e6)
# DDR4 memory Internal Vref
self.add_platform_command("set_property INTERNAL_VREF 0.84 [get_iobanks 64]")
self.add_platform_command("set_property INTERNAL_VREF 0.84 [get_iobanks 65]")
self.add_platform_command("set_property INTERNAL_VREF 0.84 [get_iobanks 66]")
# For HBM2 IP in Vivado 2019.2 (https://www.xilinx.com/support/answers/72607.html)
self.add_platform_command("connect_debug_port dbg_hub/clk [get_nets apb_clk]")

View file

@ -0,0 +1,142 @@
#!/usr/bin/env python3
#
# This file is part of LiteX-Boards.
#
# Copyright (c) 2020 Fei Gao <feig@princeton.edu>
# Copyright (c) 2020 Florent Kermarrec <florent@enjoy-digital.fr>
# Copyright (c) 2022 Jiajie Chen <c@jia.je>
# SPDX-License-Identifier: BSD-2-Clause
from migen import *
from migen.genlib.resetsync import AsyncResetSynchronizer
from litex.gen import *
from litex_boards.platforms import xilinx_vcu128
from litex.soc.cores.clock import *
from litex.soc.cores.ram.xilinx_usp_hbm2 import USPHBM2
from litex.soc.interconnect.axi import *
from litex.soc.integration.soc_core import *
from litex.soc.integration.soc import SoCRegion
from litex.soc.integration.builder import *
from litex.soc.cores.led import LedChaser
from litedram.modules import MT40A512M16
from litedram.phy import usddrphy
# CRG ----------------------------------------------------------------------------------------------
class _CRG(LiteXModule):
def __init__(self, platform, sys_clk_freq, with_hbm):
self.rst = Signal()
self.cd_sys = ClockDomain()
if with_hbm:
self.cd_hbm_ref = ClockDomain()
self.cd_apb = ClockDomain()
else: # DDR4
self.cd_sys4x = ClockDomain()
self.cd_pll4x = ClockDomain()
self.cd_idelay = ClockDomain()
# # #
self.pll = pll = USMMCM(speedgrade=-2)
self.comb += pll.reset.eq(platform.request("cpu_reset") | self.rst)
pll.register_clkin(platform.request("clk100_ddr4"), 100e6)
if with_hbm:
pll.create_clkout(self.cd_sys, sys_clk_freq)
pll.create_clkout(self.cd_hbm_ref, 100e6)
pll.create_clkout(self.cd_apb, 100e6)
platform.add_false_path_constraints(self.cd_sys.clk, self.cd_apb.clk)
else:
pll.create_clkout(self.cd_pll4x, sys_clk_freq*4, buf=None, with_reset=False)
pll.create_clkout(self.cd_idelay, 500e6)
platform.add_false_path_constraints(self.cd_sys.clk, pll.clkin) # Ignore sys_clk to pll.clkin path created by SoC's rst.
self.specials += [
Instance("BUFGCE_DIV",
p_BUFGCE_DIVIDE=4,
i_CE=1, i_I=self.cd_pll4x.clk, o_O=self.cd_sys.clk),
Instance("BUFGCE",
i_CE=1, i_I=self.cd_pll4x.clk, o_O=self.cd_sys4x.clk),
]
self.idelayctrl = USIDELAYCTRL(cd_ref=self.cd_idelay, cd_sys=self.cd_sys)
# BaseSoC ------------------------------------------------------------------------------------------
class BaseSoC(SoCCore):
def __init__(self, sys_clk_freq=125e6, with_led_chaser=True, with_hbm=False, **kwargs):
platform = xilinx_vcu128.Platform()
# CRG --------------------------------------------------------------------------------------
self.crg = _CRG(platform, sys_clk_freq, with_hbm)
# SoCCore ----------------------------------------------------------------------------------
SoCCore.__init__(self, platform, sys_clk_freq, ident="LiteX SoC on VCU128", **kwargs)
# HBM --------------------------------------------------------------------------------------
if with_hbm:
# Add HBM Core.
self.hbm = hbm = ClockDomainsRenamer({"axi": "sys"})(USPHBM2(platform))
# Get HBM .xci.
os.system("wget https://github.com/litex-hub/litex-boards/files/6893157/hbm_0.xci.txt")
os.makedirs("ip/hbm", exist_ok=True)
os.system("mv hbm_0.xci.txt ip/hbm/hbm_0.xci")
# Connect four of the HBM's AXI interfaces to the main bus of the SoC.
for i in range(4):
axi_hbm = hbm.axi[i]
axi_lite_hbm = AXILiteInterface(data_width=256, address_width=33)
self.submodules += AXILite2AXI(axi_lite_hbm, axi_hbm)
self.bus.add_slave(f"hbm{i}", axi_lite_hbm, SoCRegion(origin=0x4000_0000 + 0x1000_0000*i, size=0x1000_0000)) # 256MB.
# Link HBM2 channel 0 as main RAM
self.bus.add_region("main_ram", SoCRegion(origin=0x4000_0000, size=0x1000_0000, linker=True)) # 256MB.
elif not self.integrated_main_ram_size:
# DDR4 SDRAM -------------------------------------------------------------------------------
self.ddrphy = usddrphy.USPDDRPHY(
pads = platform.request("ddram"),
memtype = "DDR4",
sys_clk_freq = sys_clk_freq,
is_clam_shell = True,
iodelay_clk_freq = 500e6)
self.add_sdram("sdram",
phy = self.ddrphy,
module = MT40A512M16(sys_clk_freq, "1:4"),
size = 0x40000000,
l2_cache_size = kwargs.get("l2_size", 8192)
)
# Leds -------------------------------------------------------------------------------------
if with_led_chaser:
self.leds = LedChaser(
pads = platform.request_all("user_led"),
sys_clk_freq = sys_clk_freq)
# Build --------------------------------------------------------------------------------------------
def main():
from litex.build.parser import LiteXArgumentParser
parser = LiteXArgumentParser(platform=xilinx_vcu128.Platform, description="LiteX SoC on VCU128.")
parser.add_target_argument("--sys-clk-freq", default=125e6, type=float, help="System clock frequency.")
parser.add_target_argument("--with-hbm", action="store_true", help="Use HBM2.")
args = parser.parse_args()
soc = BaseSoC(
sys_clk_freq = args.sys_clk_freq,
with_hbm = args.with_hbm,
**parser.soc_argdict
)
builder = Builder(soc, **parser.builder_argdict)
if args.build:
builder.build(**parser.toolchain_argdict)
if args.load:
prog = soc.platform.create_programmer()
prog.load_bitstream(builder.get_bitstream_filename(mode="sram"))
if __name__ == "__main__":
main()