From 23c33cfa99943cfbd7eaa979f658381f57b40919 Mon Sep 17 00:00:00 2001 From: Florent Kermarrec Date: Fri, 6 Dec 2019 15:41:15 +0100 Subject: [PATCH] build: automatically add keep attribute to signals with timing constraints. Avoid having to specify it manually or eventually forget to do it and have a constraints that is not applied correctly. --- litex/build/altera/quartus.py | 3 +++ litex/build/lattice/diamond.py | 1 + litex/build/lattice/icestorm.py | 2 +- litex/build/microsemi/platform.py | 3 +++ litex/build/xilinx/ise.py | 3 +++ litex/build/xilinx/vivado.py | 3 +++ 6 files changed, 14 insertions(+), 1 deletion(-) diff --git a/litex/build/altera/quartus.py b/litex/build/altera/quartus.py index ae049a8d1..e8d01c909 100644 --- a/litex/build/altera/quartus.py +++ b/litex/build/altera/quartus.py @@ -225,11 +225,14 @@ class AlteraQuartusToolchain: return v_output.ns def add_period_constraint(self, platform, clk, period): + clk.attr.add("keep") if clk in self.clocks: raise ValueError("A period constraint already exists") period = math.floor(period*1e3)/1e3 # Round to lowest picosecond self.clocks[clk] = period def add_false_path_constraint(self, platform, from_, to): + from_.attr.add("keep") + to.attr.add("keep") if (to, from_) not in self.false_paths: self.false_paths.add((from_, to)) diff --git a/litex/build/lattice/diamond.py b/litex/build/lattice/diamond.py index b8aefc80e..dc3c5146c 100644 --- a/litex/build/lattice/diamond.py +++ b/litex/build/lattice/diamond.py @@ -194,6 +194,7 @@ class LatticeDiamondToolchain: return v_output.ns def add_period_constraint(self, platform, clk, period): + clk.attr.add("keep") # TODO: handle differential clk platform.add_platform_command("""FREQUENCY PORT "{clk}" {freq} MHz;""".format( freq=str(float(1/period)*1000), clk="{clk}"), clk=clk) diff --git a/litex/build/lattice/icestorm.py b/litex/build/lattice/icestorm.py index 0be9278fa..e59078508 100644 --- a/litex/build/lattice/icestorm.py +++ b/litex/build/lattice/icestorm.py @@ -231,8 +231,8 @@ class LatticeIceStormToolchain: return "\n".join(read_files) def add_period_constraint(self, platform, clk, period): + clk.attr.add("keep") new_freq = 1000.0/period - if clk not in self.freq_constraints.keys(): self.freq_constraints[clk] = new_freq else: diff --git a/litex/build/microsemi/platform.py b/litex/build/microsemi/platform.py index 3f7a1396b..65d2c996d 100644 --- a/litex/build/microsemi/platform.py +++ b/litex/build/microsemi/platform.py @@ -28,6 +28,7 @@ class MicrosemiPlatform(GenericPlatform): return self.toolchain.build(self, *args, **kwargs) def add_period_constraint(self, clk, period): + clk.attr.add("keep") if hasattr(clk, "p"): clk = clk.p self.toolchain.add_period_constraint(self, clk, period) @@ -37,4 +38,6 @@ class MicrosemiPlatform(GenericPlatform): from_ = from_.p if hasattr(to, "p"): to = to.p + from_.attr.add("keep") + to.attr.add("keep") self.toolchain.add_false_path_constraint(self, from_, to) diff --git a/litex/build/xilinx/ise.py b/litex/build/xilinx/ise.py index b441469c6..3a03107ec 100644 --- a/litex/build/xilinx/ise.py +++ b/litex/build/xilinx/ise.py @@ -249,6 +249,7 @@ class XilinxISEToolchain: # them through clock objects like DCM and PLL objects. def add_period_constraint(self, platform, clk, period): + clk.attr.add("keep") platform.add_platform_command( """ NET "{clk}" TNM_NET = "PRD{clk}"; @@ -258,6 +259,8 @@ TIMESPEC "TS{clk}" = PERIOD "PRD{clk}" """ + str(period) + """ ns HIGH 50%; ) def add_false_path_constraint(self, platform, from_, to): + from_.attr.add("keep") + to.attr.add("keep") platform.add_platform_command( """ NET "{from_}" TNM_NET = "TIG{from_}"; diff --git a/litex/build/xilinx/vivado.py b/litex/build/xilinx/vivado.py index aa3dca20b..fb87bccbe 100644 --- a/litex/build/xilinx/vivado.py +++ b/litex/build/xilinx/vivado.py @@ -269,12 +269,15 @@ class XilinxVivadoToolchain: return v_output.ns def add_period_constraint(self, platform, clk, period): + clk.attr.add("keep") if clk in self.clocks: raise ValueError("A period constraint already exists") period = math.floor(period*1e3)/1e3 # round to lowest picosecond self.clocks[clk] = period def add_false_path_constraint(self, platform, from_, to): + from_.attr.add("keep") + to.attr.add("keep") if (to, from_) not in self.false_paths: self.false_paths.add((from_, to))