时钟周期约束详细介绍

news/2024/12/2 14:34:53/

文章目录

      • 1.3 时钟周期约束
        • **1. create_clock**
        • **2. create_generated_clock**
        • **3. set_clock_groups**
        • **4. 创建虚拟时钟**

本文摘抄自:
个人网站:http://www.technomania.cn/
微信公众号:Quant_times、Reading_Times

1.3 时钟周期约束

时钟周期约束,顾名思义,就是我们对时钟的周期进行约束,这个约束是我们用的最多的约束了,也是最重要的约束。
下面我们讲一些vivado中时钟约束指令。

1. create_clock

在vivado中使用create_clock来创建时钟周期约束,使用方法为:

create_clock -name <name> -period <period> -waveform {<rise_time><fall_time>} [get_ports name]

在这里插入图片描述
这里的时钟必须是主时钟primary clock,主时钟通常有两种情形:一种是时钟由外部时钟源提供,通过时钟引脚引入FPGA,该时钟引脚绑定的时钟为主时钟:另一种是高速收发器(GT)的时钟RXOUTCLK或TXOUTCLK。对于7系列FPGA,需要对GT的这两个时钟收工约束:对于UltraScale FPGA,只需要对GT的输入时钟约束即可,Vivado会自动对这两个时钟约束。
如何确定主时钟是时钟周期约束的关键,除了根据主时钟的两种情形判断之外,还可以借助Tcl脚本判断。
在vivado自带的example project里面,打开CPU(HDL)工程,如下图所示。
在这里插入图片描述
把工程中的xdc文件中,create_clock的几项都注释掉。这里解释下端口(Port)和管脚(Pin)。get_ports获取的是FPGA的IO端口,get_pins获取的是FPGA内部子模块的Pin。
在这里插入图片描述
再Open Synthesized Design或者 Open Implementation Design,并通过以下两种方式查看时钟。
a. 方式一
运行tcl指令 report_clock_networks -name mainclock,显示结果如下:
在这里插入图片描述
b. 方式二
运行tcl指令 check_timing -override_defaults no_clock,显示结果如下:
在这里插入图片描述
对于高速收发器的时钟,我们也以vivado中的CPU example工程为例,看下xilinx官方约束是怎么约束的。

#Define the clocks for the GTX blocks
create_clock -name gt0_txuserclk_i -period 12.8 [get_pins mgtEngine/ROCKETIO_WRAPPER_TILE_i/gt0_ROCKETIO_WRAPPER_TILE_i/gtxe2_i/TXOUTCLK]
create_clock -name gt2_txuserclk_i -period 12.8 [get_pins mgtEngine/ROCKETIO_WRAPPER_TILE_i/gt2_ROCKETIO_WRAPPER_TILE_i/gtxe2_i/TXOUTCLK]
create_clock -name gt4_txuserclk_i -period 12.8 [get_pins mgtEngine/ROCKETIO_WRAPPER_TILE_i/gt4_ROCKETIO_WRAPPER_TILE_i/gtxe2_i/TXOUTCLK]
create_clock -name gt6_txuserclk_i -period 12.8 [get_pins mgtEngine/ROCKETIO_WRAPPER_TILE_i/gt6_ROCKETIO_WRAPPER_TILE_i/gtxe2_i/TXOUTCLK]

当系统中有多个主时钟,且这几个主时钟之间存在确定的相位关系时,需要用到-waveform参数。如果有两个主时钟,如下图所示。
在这里插入图片描述
则时钟约束为:
在这里插入图片描述
约束中数字的单位默认是ns,若不写waveform参数,则默认是占空比为50%且第一个上升沿出现在0时刻。使用report_clocks指令可以查看约束是否生效。还是上面的CPU的例子,把约束都还原到最初的状态,执行report_clock后,如下图所示,我们只列出其中几项内容。
在这里插入图片描述
一般来讲,我们输入时钟都是差分的,此时我们只对P端进行约束即可。如果同时约束了P端和N端,通过report_clock_interaction可以看到提示unsafe。这样既会增加内存开销,也会延长编译时间。

2. create_generated_clock

使用方法为:
在这里插入图片描述
在这里插入图片描述
从名字就能看出来,这个是约束我们在FPGA内部产生的衍生时钟,所以参数中有个-source,就是指定这个时钟从哪里来,这个时钟叫做master clock,是指上级时钟,区别于primary clock。它可以是我们上面讲的primary clock,也可以是其他的衍生时钟。该命令不是设定周期或波形,而是描述时钟电路如何对上级时钟进行转换。这种转换可以是下面的关系:
简单的频率分频
简单的频率倍频
频率倍频与分频的组合,获得一个非整数的比例,通常由MMCM或PLL完成
相移或波形反向
占空比改变
上述所有关系的组合
衍生时钟又分为两种情况:
a. Vivado自动推导的衍生时钟
b. 用户自定义的衍生时钟
首先来看第一种,如果使用PLL或MMCM,则Vivado会自动推导出一个约束。大家可以打开Vivado中有个叫wavegen的工程,在这个工程中,输入时钟经过PLL输出两个时钟,如下图所示。
在这里插入图片描述
但在xdc文件中,并未对这2个输出时钟进行约束,只对输入的时钟进行了约束,若我们使用report_clocks指令,则会看到:
在这里插入图片描述
注:有三个约束是因为PLL会自动输出一个反馈时钟
自动推导的好处在于当MMCM/PLL/BUFR的配置改变而影响到输出时钟的频率和相位时,用户无需改写约束,Vivado仍然可以自动推导出正确的频率/相位信息。劣势在于,用户并不清楚自动推导出的衍生时钟的名字,当设计层次改变时,衍生时钟的名字也可能改变。但由于该衍生时钟的约束并非我们自定义的,因此可能会没有关注到它名字的改变,当我们使用这些衍生时钟进行别的约束时,就会出现错误。
解决办法就是用户自己手动写出自动推导的衍生时钟的名字,也仅仅写出名字即可,其余的不写。如下如所示。
在这里插入图片描述
这一步很容易会被提示critical warning,其实有个很简单的方法,就是name 和source都按照Vivado中生成的来。

3. set_clock_groups

使用方法:
在这里插入图片描述
这个约束常用的方法有三种,第一种用法是当两个主时钟是异步关系时,使用asynchronous来指定。这个在我们平时用的还是比较多的,一般稍微大点的工程,都会出现至少两个主时钟,而且这两个时钟之间并没有任何的相位关系,这时就要指定:
在这里插入图片描述
第二种用法是当我们需要验证同一个时钟端口在不同的时钟频率下是否能获得时序收敛时使用。比如有两个异步主时钟clk1和clk2,需要验证clk2频率为100MHz,clk1频率分别为50MHz、100MHz和200MHz下的时序收敛情况,我们就可以这样写。
在这里插入图片描述
第三种用法就是当我们使用BUFGMUX时,会有两个输入时钟,但只有一一个时钟会被使用。比如MMCM输入100MHz时钟,两个输出分别为50MHz和200MHz,这两个时钟进入了BUFGMUX,如下图所示。
在这里插入图片描述
在这种情况下,我们需要设置的时序约束如下:
在这里插入图片描述

4. 创建虚拟时钟

虚拟时钟通常用于设定对输入和输出的延迟约束,这个约束其实是属于IO约束中的延迟约束,之所以放到这里来讲,是因为虚拟时钟的创建,用到了本章节讲的一些理论。虚拟时钟和前面讲的延迟约束的使用场景不太相同。顾名思义,虚拟时钟,就是没有与之绑定的物理管脚。虚拟时钟主要用于以下三个场景:
a. 外部IO的参考时钟并不是设计中的时钟
b. FPGA I/O路径参考时钟来源于内部衍生时钟,但与主时钟的频率关系并不是整数倍
c. 针对I/O指定不同的jitter和latency
简而言之,之所以要创建虚拟时钟,对于输入来说,是因为输入到FPGA数据的捕获时钟是FPGA内部产生的,与主时钟频率不同;或者PCB上有Clock Buffer导致时钟延迟不同。对于输出来说,下游器件只接收到FPGA发送过去的数据,并没有随路时钟,用自己内部的时钟去捕获数据。
如下图所示,在FPGA的A端和B端口分别有两个输入,其中捕获A端口的时钟是主时钟,而捕获B端口的时钟是MMCM输出的衍生时钟,而且该衍生时钟与主时钟的频率不是整数倍关系。
在这里插入图片描述
这种情况下时序约束如下:
在这里插入图片描述
可以看到,创建虚拟时钟用到的也是create_clock约束,但后面并没有加get_ports参数,因此被称为虚拟时钟。
再举个例子,我们常用UART和SPI,当FPGA通过串口向下游器件发送数据时,仅仅发送过去了uart_tx这个数据,下游器件通过自己内部的时钟去捕获uart_tx上的数据,这就需要通过虚拟时钟来约束;而当FPGA通过SPI向下游器件发送数据时,会发送sclk/sda/csn三个信号,其中sclk就是sda得随路时钟,下游器件通过sclk去捕获sda的数据,而不是用自己内部的时钟,这就不需要虚拟时钟,直接使用set_output_delay即可。
注意,虚拟时钟必须在约束I/O延迟之前被定义。
5. 最大最小延迟约束
顾名思义,就是设置路径的max/min delay,主要应用场景有两个:
a. 输入管脚的信号经过组合逻辑后直接输出到管脚
b. 异步电之间的最大最小延迟
在这里插入图片描述
设置方式为:

set_max_delay <delay> [-datapath_only] [-from <nodu_list>][-to <node_list>][-throurh <node_list>]
set_min_delay <delay> [-from <nodu_list>][-to <node_list>][-throurh <node_list>]

在这里插入图片描述
max/min delay的约束平时用的相对少一些,因为在跨时钟域时,我们往往会设置asynchronous或者false_path。对于异步时钟,我们一般会通过设计来保证时效能够收敛,而不是通过时序约束来保证。


http://www.ppmy.cn/news/666685.html

相关文章

易语言使用线程时钟后MYSQL_易语言多线程时钟周期 易语言三个时钟同时运行

易语言代码&#xff0c;时钟1.时钟周期1000是什么意思&#xff1f; 时钟周期事件每秒执行一次&#xff1a;10001秒&#xff0c;1000010秒&#xff0c;依此类推。时钟停止是时钟1。时钟周期0&#xff0c;而不是时钟1。禁用真。时钟是计时器的意识。定期执行一段代码。这个间隔就…

4 FPGA时序约束理论篇之时钟周期约束

时钟周期约束 时钟周期约束&#xff0c;顾名思义&#xff0c;就是我们对时钟的周期进行约束&#xff0c;这个约束是我们用的最多的约束了&#xff0c;也是最重要的约束。 下面我们讲一些Vivado中时钟约束指令。 1. Create_clock 在Vivado中使用create_clock来创建时钟周期约束…

ARM7指令时钟周期数

转自&#xff1a;http://www.elecfans.com/emb/arm/2009071678028.html ARM7具有3级流水线结构&#xff08;取指、译码、执行&#xff09;&#xff0c;对大多数指令来说每条流水线的处理都是单周期的&#xff0c;不过某些情况下&#xff0c;取指和执行的周期数会延长&#xff…

时钟周期 机器周期 指令周期

时钟周期 时钟周期是一个时间的量&#xff0c;人们规定10纳秒&#xff08;ns&#xff09;为一个时钟周期。时钟周期表示了SDRAM所能运行的最高频率。更小的时钟周期就意味着更高的工作频率。对于PC100规格的内存来说&#xff0c;它的运行时钟周期应该不高于10纳秒。纳秒与工作频…

c语言执行一条指令几个机器周期,时钟周期、机器周期与指令周期

时钟周期、机器周期与指令周期 1、51系列单片机中&#xff0c;外部输入震荡输入经2分频后作为时钟&#xff1b;另外&#xff0c;一个机器周期由6个时钟周期组成&#xff1b;因为没有流水线(pipeline)结构&#xff0c;执行一条指令需要经过取指令、译码、存取操作数、执行、保存…

时钟周期和机器周期之间的关系

时钟周期和机器周期之间的关系 时钟周期&#xff08;小&#xff09;机器周期&#xff08;中&#xff09;指令周期 &#xff08;大&#xff09;作用 时钟周期&#xff08;小&#xff09; 时钟周期&#xff1a;单片机时钟控制的基本时间单位。以微秒(μs)作单位。 时钟周期受时…

计算机时钟周期的概念,时钟周期、机器周期、指令周期的概念及三者之间的关系...

时钟周期、机器周期、指令周期的概念及三者之间的关系 帖子创建时间: 2012年12月31日 16:49 评论&#xff1a;0 时钟周期、机器周期、指令周期的概念及三者之间的关系 时钟周期 时钟周期也称为振荡周期&#xff0c;定义为时钟脉冲的倒数(可以这样来理解&#xff0c;时钟周期就…

CPU中的指令周期、CPU周期和时钟周期

计算机中我们常常会混淆指令周期、CPU周期和时钟周期&#xff0c;要区分这些并不难&#xff0c;但要想彻底弄懂这些&#xff0c;就得要求我们对CPU底层有一定了解。 一.指令周期 指令周期&#xff1a;是指计算机从取指到指令执行完毕的时间 计算机执行指令的过程可以分为以下…