167 lines
6.6 KiB
Python
Executable File
167 lines
6.6 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
|
|
#
|
|
# This file is part of LiteX-Boards.
|
|
#
|
|
# Copyright (c) 2022 Florent Kermarrec <florent@enjoy-digital.fr>
|
|
# SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
# Build/Use:
|
|
# ./limesdr_mini_v2.py --csr-csv=csr.csv --build --load
|
|
# litex_server --jtag --jtag-config=openocd_limesdr_mini_v2.cfg
|
|
# litex_term crossover
|
|
|
|
from migen import *
|
|
|
|
from litex_boards.platforms import limesdr_mini_v2
|
|
|
|
from litex.build.lattice.trellis import trellis_args, trellis_argdict
|
|
|
|
from litex.soc.cores.clock import *
|
|
from litex.soc.interconnect.csr import *
|
|
from litex.soc.integration.soc_core import *
|
|
from litex.soc.integration.builder import *
|
|
from litex.soc.interconnect import stream
|
|
|
|
from litex.soc.cores.led import LedChaser
|
|
from litex.soc.cores.bitbang import I2CMaster
|
|
from litex.soc.cores.usb_fifo import FT245PHYSynchronous
|
|
|
|
from litescope import LiteScopeAnalyzer
|
|
|
|
# CRG ----------------------------------------------------------------------------------------------
|
|
|
|
class _CRG(Module):
|
|
def __init__(self, platform, sys_clk_freq):
|
|
self.rst = Signal()
|
|
self.clock_domains.cd_sys = ClockDomain()
|
|
self.clock_domains.cd_usb = ClockDomain()
|
|
|
|
# # #
|
|
|
|
# Clk.
|
|
clk40 = platform.request("clk40")
|
|
|
|
# PLL.
|
|
self.submodules.pll = pll = ECP5PLL()
|
|
self.comb += pll.reset.eq(self.rst)
|
|
pll.register_clkin(clk40, 40e6)
|
|
pll.create_clkout(self.cd_sys, sys_clk_freq)
|
|
|
|
# USB.
|
|
self.comb += self.cd_usb.clk.eq(platform.request("usb_fifo_clk"))
|
|
|
|
# BoardInfo ----------------------------------------------------------------------------------------
|
|
|
|
class BoardInfo(Module, AutoCSR):
|
|
def __init__(self, revision_pads):
|
|
self.revision = CSRStorage(fields=[
|
|
CSRField("hardware", size=4, description="Hardware Revision."),
|
|
CSRField("bom", size=4, description="Bill of Material Revision."),
|
|
])
|
|
|
|
# # #
|
|
|
|
self.comb += self.revision.fields.hardware.eq(revision_pads.hardware)
|
|
self.comb += self.revision.fields.bom .eq(revision_pads.bom)
|
|
|
|
# BaseSoC ------------------------------------------------------------------------------------------
|
|
|
|
class BaseSoC(SoCCore):
|
|
def __init__(self, sys_clk_freq=int(80e6), toolchain="trellis",
|
|
with_usb_fifo = True, with_usb_fifo_loopback=False,
|
|
with_led_chaser = True,
|
|
**kwargs):
|
|
platform = limesdr_mini_v2.Platform(toolchain=toolchain)
|
|
|
|
# SoCCore ----------------------------------------------------------------------------------
|
|
kwargs["uart_name"] = "crossover"
|
|
SoCCore.__init__(self, platform, sys_clk_freq, ident="LiteX SoC on LimeSDR-Mini-V2", **kwargs)
|
|
|
|
# JTAGBone ---------------------------------------------------------------------------------
|
|
self.add_jtagbone()
|
|
|
|
# CRG --------------------------------------------------------------------------------------
|
|
self.submodules.crg = _CRG(platform, sys_clk_freq)
|
|
|
|
# Info -------------------------------------------------------------------------------------
|
|
self.submodules.info = BoardInfo(platform.request("revision"))
|
|
|
|
# I2C Bus ----------------------------------------------------------------------------------
|
|
# - Temperature Sensor (LM72 @ 0x48).
|
|
# - Eeprom (M24128 @ 0x50) / Not populated.
|
|
self.submodules.i2c = I2CMaster(platform.request("i2c"))
|
|
|
|
# USB-FIFO ---------------------------------------------------------------------------------
|
|
if with_usb_fifo:
|
|
usb_pads = platform.request("usb_fifo")
|
|
self.submodules.usb_phy = usb_phy = FT245PHYSynchronous(
|
|
pads = usb_pads,
|
|
clk_freq = sys_clk_freq,
|
|
fifo_depth = 8,
|
|
read_time = 128,
|
|
write_time = 128,
|
|
)
|
|
if with_usb_fifo_loopback:
|
|
usb_loopback = stream.SyncFIFO([("data", 32)], 2048, buffered=True)
|
|
self.submodules += usb_loopback
|
|
self.comb += [
|
|
usb_phy.source.connect(usb_loopback.sink),
|
|
usb_loopback.source.connect(usb_phy.sink),
|
|
]
|
|
else:
|
|
self.comb += usb_phy.source.ready.eq(1) # Accept incoming stream to validate Host -> FPGA.
|
|
|
|
analyzer_probes = usb_phy.get_litescope_probes()
|
|
self.submodules.analyzer = LiteScopeAnalyzer(analyzer_probes,
|
|
depth = 512,
|
|
clock_domain = "usb",
|
|
samplerate = sys_clk_freq,
|
|
csr_csv = "analyzer.csv"
|
|
)
|
|
|
|
# Debug -------------------------------------------------------------------------------
|
|
egpio_pads = platform.request("egpio")
|
|
self.comb += egpio_pads[0].eq(ClockSignal("sys"))
|
|
self.comb += egpio_pads[1].eq(ClockSignal("usb"))
|
|
|
|
# Leds -------------------------------------------------------------------------------------
|
|
if with_led_chaser:
|
|
leds_g = Signal(4)
|
|
leds_r = Signal(4)
|
|
self.comb += platform.request_all("led_g_n").eq(~leds_g)
|
|
self.comb += platform.request_all("led_r_n").eq(~leds_r)
|
|
self.submodules.leds = LedChaser(Cat(leds_g, leds_r), sys_clk_freq)
|
|
|
|
# Build --------------------------------------------------------------------------------------------
|
|
|
|
def main():
|
|
from litex.soc.integration.soc import LiteXSoCArgumentParser
|
|
parser = LiteXSoCArgumentParser(description="LiteX SoC on LimeSDR-Mini-V2")
|
|
target_group = parser.add_argument_group(title="Target options")
|
|
target_group.add_argument("--build", action="store_true", help="Build design.")
|
|
target_group.add_argument("--load", action="store_true", help="Load bitstream.")
|
|
target_group.add_argument("--toolchain", default="trellis", help="FPGA toolchain (trellis or diamond).")
|
|
target_group.add_argument("--sys-clk-freq", default=80e6, help="System clock frequency.")
|
|
builder_args(parser)
|
|
soc_core_args(parser)
|
|
trellis_args(parser)
|
|
args = parser.parse_args()
|
|
|
|
soc = BaseSoC(
|
|
sys_clk_freq = int(float(args.sys_clk_freq)),
|
|
toolchain = args.toolchain,
|
|
**soc_core_argdict(args)
|
|
)
|
|
builder = Builder(soc, **builder_argdict(args))
|
|
builder_kargs = trellis_argdict(args) if args.toolchain == "trellis" else {}
|
|
if args.build:
|
|
builder.build(**builder_kargs)
|
|
|
|
if args.load:
|
|
prog = soc.platform.create_programmer()
|
|
prog.load_bitstream(builder.get_bitstream_filename(mode="sram", ext=".svf")) # FIXME
|
|
|
|
if __name__ == "__main__":
|
|
main()
|