diff --git a/litex/build/anlogic/platform.py b/litex/build/anlogic/platform.py index 334c75297a..b66f002b77 100644 --- a/litex/build/anlogic/platform.py +++ b/litex/build/anlogic/platform.py @@ -13,7 +13,8 @@ # AnlogicPlatform ---------------------------------------------------------------------------------- class AnlogicPlatform(GenericPlatform): - _bitstream_ext = ".bit" + _bitstream_ext = ".bit" + _jtag_support = False _supported_toolchains = ["td"] diff --git a/litex/build/colognechip/platform.py b/litex/build/colognechip/platform.py index 127c4e2fb8..78f6ed6e8f 100644 --- a/litex/build/colognechip/platform.py +++ b/litex/build/colognechip/platform.py @@ -13,6 +13,7 @@ class CologneChipPlatform(GenericPlatform): bitstream_ext = "_00.cfg.bit" + _jtag_support = False _supported_toolchains = ["colognechip"] diff --git a/litex/build/generic_platform.py b/litex/build/generic_platform.py index a9a24eeaae..933dd567d6 100644 --- a/litex/build/generic_platform.py +++ b/litex/build/generic_platform.py @@ -329,6 +329,7 @@ def get_platform_commands(self): class GenericPlatform: device_family = None + _jtag_support = True # JTAGBone can't be used with all FPGAs. _bitstream_ext = None # None by default, overridden by vendor platform, may # be a string when same extension is used for sram and # flash. A dict must be provided otherwise @@ -504,6 +505,16 @@ def get_bitstream_extension(self, mode="sram"): def create_programmer(self): raise NotImplementedError + @property + def jtag_support(self): + if isinstance(self._jtag_support, str): + return self._jtag_support + else: + for dev in self._jtag_support: + if self.device.startswith(dev): + return True + return False + @property def support_mixed_language(self): return self.toolchain.support_mixed_language diff --git a/litex/build/gowin/platform.py b/litex/build/gowin/platform.py index 1e7fb419c8..b694c2cc39 100644 --- a/litex/build/gowin/platform.py +++ b/litex/build/gowin/platform.py @@ -14,6 +14,7 @@ class GowinPlatform(GenericPlatform): _bitstream_ext = ".fs" + _jtag_support = False _supported_toolchains = ["gowin", "apicula"] diff --git a/litex/build/microsemi/platform.py b/litex/build/microsemi/platform.py index 9be106e04e..71c344d653 100644 --- a/litex/build/microsemi/platform.py +++ b/litex/build/microsemi/platform.py @@ -11,6 +11,7 @@ class MicrosemiPlatform(GenericPlatform): _bitstream_ext = ".bit" + _jtag_support = False _supported_toolchains = ["libero_soc_polarfire"] diff --git a/litex/build/parser.py b/litex/build/parser.py index 827524baba..ca46efe941 100644 --- a/litex/build/parser.py +++ b/litex/build/parser.py @@ -9,11 +9,14 @@ import logging import argparse import importlib +import time from litex.soc.cores import cpu from litex.soc.integration import soc_core from litex.soc.integration import builder +from litex.gen.common import * + # Litex Argument Parser ---------------------------------------------------------------------------- class LiteXArgumentParser(argparse.ArgumentParser): @@ -63,6 +66,9 @@ def __init__(self, platform=None, **kwargs): self.set_platform(platform) self.add_target_group() self.add_logging_group() + # workaround for backward compatibility + self._rm_jtagbone = False + self._rm_uartbone = False def set_platform(self, platform): """ set platform. Check first if not already set @@ -104,8 +110,21 @@ def add_target_argument(self, *args, **kwargs): """ wrapper to add argument to "Target options group" from outer of this class """ + arg = args[0] + if arg in ["--with-jtagbone", "--with-uartbone"]: + if arg == "--with-jtagbone": + self._rm_jtagbone = True + else: + self._rm_uartbone = True + print("Warning {} {} {}".format( + colorer(arg, color="red"), + colorer(" is added by SoCCore. ", color="red"), + colorer("Please remove this option from target", color="yellow"))) + time.sleep(2) + return # bypass insert if self._target_group is None: self._target_group = self.add_argument_group(title="Target options") + self._target_group.add_argument(*args, **kwargs) def add_logging_group(self): @@ -147,7 +166,14 @@ def soc_argdict(self): ====== soc_core arguments dict """ - return soc_core.soc_core_argdict(self._args) # FIXME: Rename to soc_argdict in the future. + soc_arg = soc_core.soc_core_argdict(self._args) # FIXME: Rename to soc_argdict in the future. + + # Work around for backward compatibility + if self._rm_jtagbone: + soc_arg.pop("with_jtagbone") + if self._rm_uartbone: + soc_arg.pop("with_uartbone") + return soc_arg @property def toolchain_argdict(self): diff --git a/litex/build/quicklogic/platform.py b/litex/build/quicklogic/platform.py index ce099aac18..193f45262e 100644 --- a/litex/build/quicklogic/platform.py +++ b/litex/build/quicklogic/platform.py @@ -13,6 +13,7 @@ class QuickLogicPlatform(GenericPlatform): _bitstream_ext = ".bit" + _jtag_support = False _supported_toolchains = ["f4pga"] diff --git a/litex/build/xilinx/platform.py b/litex/build/xilinx/platform.py index cfe1b2f942..810b8759cb 100644 --- a/litex/build/xilinx/platform.py +++ b/litex/build/xilinx/platform.py @@ -26,6 +26,12 @@ class XilinxPlatform(GenericPlatform): "ultrascale+" : ["vivado"], } + _jtag_support = [ + "xc6", + "xc7a", "xc7k", "xc7v", "xc7z", + "xcau", "xcku", "xcvu", "xczu" + ] + def __init__(self, *args, toolchain="ise", **kwargs): GenericPlatform.__init__(self, *args, **kwargs) self.edifs = set() @@ -126,6 +132,7 @@ def get_argdict(cls, toolchain, args): else: return dict() + # XilinxSpartan6Platform --------------------------------------------------------------------------- class XilinxSpartan6Platform(XilinxPlatform): diff --git a/litex/soc/integration/soc_core.py b/litex/soc/integration/soc_core.py index c0eb5942dc..4a9b5e10e9 100644 --- a/litex/soc/integration/soc_core.py +++ b/litex/soc/integration/soc_core.py @@ -108,6 +108,13 @@ def __init__(self, platform, clk_freq, # Controller parameters with_ctrl = True, + # JTAGBone + with_jtagbone = False, + jtagbone_chain = 1, + + # UARTBone + with_uartbone = False, + # Others **kwargs): @@ -174,6 +181,31 @@ def __init__(self, platform, clk_freq, # Wishbone Slaves. self.wb_slaves = {} + # Parameters check validity ---------------------------------------------------------------- + + # Check if jtagbone is supported (SPI only device or no user access). + if with_jtagbone: + if not platform.jtag_support: + self.logger.error("{} {} with {} FPGA".format( + colorer("JTAGBone isn't supported for platform", color="red"), + platform.name, platform.device)) + raise SoCError() + if with_uart: + # crossover+uartbone is kept as backward compatibility + if uart_name == "crossover+uartbone": + self.logger.warning("{} UART: is deprecated {}".format( + colorer(uart_name, color="yellow"), + colorer("please use --uart-name=\"crossover\" --with-uartbone", color="red"))) + time.sleep(2) + # Already configured. + self._uartbone = True + uart_name = "crossover" + + # JTAGBone and jtag_uart can't be used at the same time. + assert not (with_jtagbone and uart_name == "jtag_uart") + # UARTBone and serial can't be used at the same time. + assert not (with_uartbone and uart_name == "serial") + # Modules instances ------------------------------------------------------------------------ # Add SoCController @@ -220,10 +252,18 @@ def __init__(self, platform, clk_freq, if ident != "": self.add_identifier("identifier", identifier=ident, with_build_time=ident_version) + # Add UARTBone + if with_uartbone: + self.add_uartbone(baudrate=uart_baudrate) + # Add UART if with_uart: self.add_uart(name="uart", uart_name=uart_name, baudrate=uart_baudrate, fifo_depth=uart_fifo_depth) + # Add JTAGBone + if with_jtagbone: + self.add_jtagbone(chain=jtagbone_chain) + # Add Timer if with_timer: self.add_timer(name="timer0") @@ -294,6 +334,13 @@ def soc_core_args(parser): soc_group.add_argument("--uart-baudrate", default=115200, type=auto_int, help="UART baudrate.") soc_group.add_argument("--uart-fifo-depth", default=16, type=auto_int, help="UART FIFO depth.") + # UARTBone parameters + soc_group.add_argument("--with-uartbone", action="store_true", help="Enable UARTbone.") + + # JTAGBone parameters + soc_group.add_argument("--with-jtagbone", action="store_true", help="Enable Jtagbone support.") + soc_group.add_argument("--jtagbone-chain", default=1, type=int, help="Jtagbone chain index.") + # Timer parameters soc_group.add_argument("--no-timer", action="store_true", help="Disable Timer.") soc_group.add_argument("--timer-uptime", action="store_true", help="Add an uptime capability to Timer.")