Add embedfire_rise_pro

Add Flash
This commit is contained in:
Yu-Ti Kuo 2024-11-02 10:09:06 +08:00
parent 041c1607ce
commit d7f2b5a929
2 changed files with 356 additions and 0 deletions

View file

@ -0,0 +1,152 @@
#
# This file is part of LiteX-Boards.
#
# Copyright (c) 2024 Yu-Ti Kuo <bobgash2@gmail.com>
# SPDX-License-Identifier: BSD-2-Clause
# embedfire rise pro FPGA: https://detail.tmall.com/item.htm?id=645153441975
from litex.build.generic_platform import *
from litex.build.xilinx import Xilinx7SeriesPlatform
from litex.build.openocd import OpenOCD
from litex.build.xilinx.programmer import VivadoProgrammer
# IOs ----------------------------------------------------------------------------------------------
_io = [
# Clk / Rst
("clk50" , 0, Pins("W19"), IOStandard("LVCMOS33")),
("cpu_reset", 0, Pins("N15"), IOStandard("LVCMOS33")),
# Leds
("user_led", 0, Pins("M21"), IOStandard("LVCMOS33")),
("user_led", 1, Pins("L21"), IOStandard("LVCMOS33")),
("user_led", 2, Pins("K21"), IOStandard("LVCMOS33")),
("user_led", 3, Pins("K22"), IOStandard("LVCMOS33")),
# Buttons
("user_sw", 0, Pins("V17"), IOStandard("LVCMOS33")),
("user_sw", 1, Pins("W17"), IOStandard("LVCMOS33")),
("user_sw", 2, Pins("AA18"), IOStandard("LVCMOS33")),
("user_sw", 3, Pins("AB18"), IOStandard("LVCMOS33")),
# Beeper (Buzzer)
("beeper", 0, Pins("M17"), IOStandard("LVCMOS33")),
# Fan
("fan", 0, Pins("W22"), IOStandard("LVCMOS33")),
# Serial CH340G
("serial", 0,
Subsignal("tx", Pins("N17")),
Subsignal("rx", Pins("P17")),
IOStandard("LVCMOS33")
),
# I2C EEPROM 24C64
("i2c", 0,
Subsignal("scl", Pins("E22")),
Subsignal("sda", Pins("D22")),
IOStandard("LVCMOS33"),
),
# DDR3 SDRAM MT41K256M16
("ddram", 0,
Subsignal("a", Pins(
"AA4 AB2 AA5 AB3 AB1 U2 W1 R2",
"V2 U3 Y1 W2 Y2 U1 V3"),
IOStandard("SSTL135")),
Subsignal("ba", Pins("AA1 Y3 AA3"), IOStandard("SSTL135")),
Subsignal("ras_n", Pins("W6"), IOStandard("SSTL135")),
Subsignal("cas_n", Pins("U5"), IOStandard("SSTL135")),
Subsignal("we_n", Pins("Y4"), IOStandard("SSTL135")),
Subsignal("cs_n", Pins("T1"), IOStandard("SSTL135")),
Subsignal("dm", Pins("D2 G2 M2 M5"), IOStandard("SSTL135")),
Subsignal("dq", Pins(
"C2 G1 A1 F3 B2 F1 B1 E2",
"H3 G3 H2 H5 J1 J5 K1 H4",
"L4 M3 L3 J6 K3 K6 J4 L5",
"P1 N4 R1 N2 M6 N5 P6 P2"),
IOStandard("SSTL135"),
Misc("IN_TERM=UNTUNED_SPLIT_40")),
Subsignal("dqs_p", Pins("E1 K2 M1 P5"),
IOStandard("DIFF_SSTL135"),
Misc("IN_TERM=UNTUNED_SPLIT_40")),
Subsignal("dqs_n", Pins("D1 J2 L1 P4"),
IOStandard("DIFF_SSTL135"),
Misc("IN_TERM=UNTUNED_SPLIT_40")),
Subsignal("clk_p", Pins("V4"), IOStandard("DIFF_SSTL135")),
Subsignal("clk_n", Pins("W4"), IOStandard("DIFF_SSTL135")),
Subsignal("cke", Pins("AB5"), IOStandard("SSTL135")),
Subsignal("odt", Pins("T5"), IOStandard("SSTL135")),
Subsignal("reset_n", Pins("R3"), IOStandard("SSTL135")),
Misc("SLEW=FAST"),
),
# RGMII Ethernet (RTL8211F)
("eth_clocks", 0,
Subsignal("tx", Pins("C18")),
Subsignal("rx", Pins("C19")),
IOStandard("LVCMOS33")
),
("eth", 0,
#SubSignal("inib"), Pins("D21")),
#Subsignal("rst_n", Pins("E21")),
Subsignal("mdio", Pins("G22")),
Subsignal("mdc", Pins("G21")),
Subsignal("rx_ctl", Pins("C22")),
Subsignal("rx_data", Pins("D20 C20 A18 A19")),
Subsignal("tx_ctl", Pins("B22")),
Subsignal("tx_data", Pins("B20 A20 B21 A21")),
IOStandard("LVCMOS33")
),
# SDCard
("spisdcard", 0,
Subsignal("cd", Pins("AA19")),
Subsignal("clk", Pins("Y22")),
Subsignal("mosi", Pins("Y21")),
Subsignal("cs_n", Pins("A14")),
Subsignal("miso", Pins("AB21")),
Misc("SLEW=FAST"),
IOStandard("LVCMOS33"),
),
("sdcard", 0,
Subsignal("data", Pins("AB21 AB22 AB20 W21"),),
Subsignal("cmd", Pins("Y21"),),
Subsignal("clk", Pins("Y22")),
Subsignal("cd", Pins("AA19")),
Misc("SLEW=FAST"),
IOStandard("LVCMOS33"),
),
]
# Connectors ---------------------------------------------------------------------------------------
_connectors = [] # ToDo
# Platform -----------------------------------------------------------------------------------------
class Platform(Xilinx7SeriesPlatform):
default_clk_name = "clk50"
default_clk_period = 1e9/50e6
def __init__(self, variant="a7-35", toolchain="vivado"):
device = {
"a7-35": "xc7a35tfgg484-2",
"a7-100": "xc7a100tfgg484-2",
"a7-200": "xc7a200tfbg484-2"
}[variant]
Xilinx7SeriesPlatform.__init__(self, device, _io, _connectors, toolchain=toolchain)
self.toolchain.bitstream_commands = \
["set_property BITSTREAM.CONFIG.SPI_BUSWIDTH 4 [current_design]"]
self.toolchain.additional_commands = \
["write_cfgmem -force -format bin -interface spix4 -size 16 "
"-loadbit \"up 0x0 {build_name}.bit\" -file {build_name}.bin"]
self.add_platform_command("set_property INTERNAL_VREF 0.675 [get_iobanks 34]")
def create_programmer(self):
return VivadoProgrammer(flash_part="mt25ql128-spi-x1_x2_x4")
def do_finalize(self, fragment):
Xilinx7SeriesPlatform.do_finalize(self, fragment)
self.add_period_constraint(self.lookup_request("clk50", loose=True), 1e9/50e6)

View file

@ -0,0 +1,204 @@
#!/usr/bin/env python3
#
# This file is part of LiteX-Boards.
#
# Copyright (c) 2024 Yu-Ti Kuo <bobgash2@gmail.com>
# SPDX-License-Identifier: BSD-2-Clause
# embedfire rise pro FPGA: https://detail.tmall.com/item.htm?id=645153441975
from migen import *
from litex.gen import *
from litex_boards.platforms import embedfire_rise_pro
from litex.soc.cores.clock import *
from litex.soc.integration.soc import SoCRegion
from litex.soc.integration.soc_core import *
from litex.soc.integration.builder import *
from litex.soc.cores.led import LedChaser
from litex.soc.cores.gpio import GPIOIn
from litex.soc.cores.xadc import XADC
from litex.soc.cores.dna import DNA
from litex.soc.cores.pwm import PWM
from litedram.modules import MT41K256M16
from litedram.phy import s7ddrphy
from liteeth.phy.mii import LiteEthPHYMII
# CRG ----------------------------------------------------------------------------------------------
class _CRG(LiteXModule):
def __init__(self, platform, sys_clk_freq, with_dram=True, with_rst=True):
self.rst = Signal()
self.cd_sys = ClockDomain()
if with_dram:
self.cd_sys4x = ClockDomain()
self.cd_sys4x_dqs = ClockDomain()
self.cd_idelay = ClockDomain()
# # #
# Clk/Rst.
clk50 = platform.request("clk50")
rst = ~platform.request("cpu_reset") if with_rst else 0
# PLL.
self.pll = pll = S7PLL(speedgrade=-1)
self.comb += pll.reset.eq(rst | self.rst)
pll.register_clkin(clk50, 50e6)
pll.create_clkout(self.cd_sys, sys_clk_freq)
platform.add_false_path_constraints(self.cd_sys.clk, pll.clkin) # Ignore sys_clk to pll.clkin path created by SoC's rst.
if with_dram:
pll.create_clkout(self.cd_sys4x, 4*sys_clk_freq)
pll.create_clkout(self.cd_sys4x_dqs, 4*sys_clk_freq, phase=90)
pll.create_clkout(self.cd_idelay, 200e6)
# IdelayCtrl.
if with_dram:
self.idelayctrl = S7IDELAYCTRL(self.cd_idelay)
# BaseSoC ------------------------------------------------------------------------------------------
class BaseSoC(SoCCore):
def __init__(self, variant="a7-35", toolchain="vivado", sys_clk_freq=50e6,
with_xadc = False,
with_dna = False,
with_ethernet = False,
with_etherbone = False,
eth_ip = "192.168.1.50",
remote_ip = None,
eth_dynamic_ip = False,
with_led_chaser = True,
with_buttons = False,
with_beeper = True,
**kwargs):
platform = embedfire_rise_pro.Platform(variant=variant, toolchain=toolchain)
# CRG --------------------------------------------------------------------------------------
with_dram = (kwargs.get("integrated_main_ram_size", 0) == 0)
self.crg = _CRG(platform, sys_clk_freq, with_dram)
# SoCCore ----------------------------------------------------------------------------------
SoCCore.__init__(self, platform, sys_clk_freq, ident="LiteX SoC on embedfire", **kwargs)
# XADC -------------------------------------------------------------------------------------
if with_xadc:
self.xadc = XADC()
# DNA --------------------------------------------------------------------------------------
if with_dna:
self.dna = DNA()
self.dna.add_timing_constraints(platform, sys_clk_freq, self.crg.cd_sys.clk)
# DDR3 SDRAM -------------------------------------------------------------------------------
if not self.integrated_main_ram_size:
self.ddrphy = s7ddrphy.A7DDRPHY(platform.request("ddram"),
memtype = "DDR3",
nphases = 4,
sys_clk_freq = sys_clk_freq)
self.add_sdram("sdram",
phy = self.ddrphy,
module = MT41K256M16(sys_clk_freq, "1:4"),
l2_cache_size = kwargs.get("l2_size", 8192)
)
# Ethernet / Etherbone ---------------------------------------------------------------------
if with_ethernet or with_etherbone:
self.ethphy = LiteEthPHYMII(
clock_pads = self.platform.request("eth_clocks"),
pads = self.platform.request("eth"))
if with_etherbone:
self.add_etherbone(phy=self.ethphy, ip_address=eth_ip, with_ethmac=with_ethernet)
elif with_ethernet:
self.add_ethernet(phy=self.ethphy, dynamic_ip=eth_dynamic_ip, local_ip=eth_ip, remote_ip=remote_ip)
# Leds -------------------------------------------------------------------------------------
if with_led_chaser:
self.leds = LedChaser(
pads = platform.request_all("user_led"),
sys_clk_freq = sys_clk_freq,
)
self.leds.add_pwm()
# Buttons ----------------------------------------------------------------------------------
if with_buttons:
self.buttons = GPIOIn(
pads = platform.request_all("user_btn"),
with_irq = self.irq.enabled
)
# Beeper------------------------------------------------------------------------------------
self.beeper = PWM(
pwm=platform.request("beeper", 0),
with_csr = True,
default_enable = False,
default_width = 0x800,
default_period = 0xfff,
)
self.fan_pwm = PWM(
pwm=platform.request("fan", 0),
with_csr = True,
default_enable = False,
default_width = 0x800,
default_period = 0xfff,
)
# Build --------------------------------------------------------------------------------------------
def main():
from litex.build.parser import LiteXArgumentParser
parser = LiteXArgumentParser(platform=embedfire_rise_pro.Platform, description="LiteX SoC on embedfire rise pro.")
parser.add_target_argument("--flash", action="store_true", help="Flash bitstream.")
parser.add_target_argument("--variant", default="a7-35", help="Board variant (a7-35 or a7-100 or a7-200).")
parser.add_target_argument("--sys-clk-freq", default=50e6, type=float, help="System clock frequency.")
parser.add_target_argument("--with-xadc", action="store_true", help="Enable 7-Series XADC.")
parser.add_target_argument("--with-dna", action="store_true", help="Enable 7-Series DNA.")
parser.add_target_argument("--with-ethernet", action="store_true", help="Enable Ethernet support.")
parser.add_target_argument("--with-etherbone", action="store_true", help="Enable Etherbone support.")
parser.add_target_argument("--eth-ip", default="192.168.1.50", help="Ethernet/Etherbone IP address.")
parser.add_target_argument("--remote-ip", default="192.168.1.100", help="Remote IP address of TFTP server.")
parser.add_target_argument("--eth-dynamic-ip", action="store_true", help="Enable dynamic Ethernet IP addresses setting.")
sdopts = parser.target_group.add_mutually_exclusive_group()
sdopts.add_argument("--with-spi-sdcard", action="store_true", help="Enable SPI-mode SDCard support.")
sdopts.add_argument("--with-sdcard", action="store_true", help="Enable SDCard support.")
args = parser.parse_args()
assert not (args.with_etherbone and args.eth_dynamic_ip)
soc = BaseSoC(
variant = args.variant,
toolchain = args.toolchain,
sys_clk_freq = args.sys_clk_freq,
with_xadc = args.with_xadc,
with_dna = args.with_dna,
with_ethernet = args.with_ethernet,
with_etherbone = args.with_etherbone,
eth_ip = args.eth_ip,
remote_ip = args.remote_ip,
eth_dynamic_ip = args.eth_dynamic_ip,
**parser.soc_argdict
)
if args.with_spi_sdcard:
soc.add_spi_sdcard()
if args.with_sdcard:
soc.add_sdcard()
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 args.flash:
prog = soc.platform.create_programmer()
prog.flash(0, builder.get_bitstream_filename(mode="flash"))
if __name__ == "__main__":
main()