FPGA管脚分配时需注意的一些事项(以xilinx xc4vsx55为例)
FPGA管脚分配时需注意的一些事项(以xilinx xc4vsx55为例) 平台:XC4VSX55 ISE10.1
设计过FPGA的原理图,看FPGA的手册,说管脚的分配问题,如时钟管脚要用GC类管脚,而且单端时钟输入时要用P类型的管脚,不能用N类型管脚等等。 一直以来都没有试验过,今天试验一把,以求各种验证。 1)GC类全局时钟管脚是否可用作普通IO使用?
所谓GC类管脚,就是在管脚的称是诸如IO_L1P_GC_LC等带有GC的管脚。其实手册中说的是GC类管脚可以用作IO的,但在《Xilinx FPGA开发实用教程》(清华出版社)574页倒数第八行提到:“所有从全局时钟管脚输入的信号必须经过IBUF元,否则在布局布线时会报错”,于是今天我试了一下,将某一GC类管脚分配给一个普通的输入口(也试验了分配给一个普通的输出口),经布局布线后,未出错。 因此得出结论:GC类全局时钟管脚可以作为普通IO使用。(不知道是不是我对书中提到的全局时钟管脚理解有误,如果是,请网友别拍我,敬请留言指正) 2)非GC类全局时钟管脚是否可以作时钟使用?
其实至于说能否作为时钟使用,这里有另一层函义。当然,如果你把一个普通IO口配置成输入口,就把它的输入信号作为时钟,那是没问题的。但我们一般不这么做,因为时钟信号对于我们来说是一个很重要的信号,因此FPGA在内部会有特殊照顾,如果你使用FPGA传门为时钟预留的管脚,并作一些处理,那么你的时钟对于各种模块的时延是可以忽略的,因为时钟在布线时是单独走的一层,而如果你就仅用普通IO的话,经过FPGA内部布局布线后,从它的输到,再到各个使用时钟的地方,有的线长,有的线短,它的时延将是不一样的。这些东西还是看一些FPGA结构的内容吧。
在xilinx里有专门的DCM IP核可供调用,在ISE中执行project——>New Source——>IP(CORE Generator & Architecture Wizard)——>FPGA Features and
Design——>Clocking——>Virtex-4——>Single DCM ADV v9.1i,可得如下界面:
需要特别注意的是CLKIN Source需要选择是External还是Internal,各自生成的源文件如下: ==========================选择External========================= `timescale 1ns / 1ps
module clk_test(CLKIN_IN, CLKIN_IBUFG_OUT, CLK0_OUT, LOCKED_OUT); input CLKIN_IN;
output CLKIN_IBUFG_OUT; output CLK0_OUT; output LOCKED_OUT;
wire CLKFB_IN; wire CLKIN_IBUFG; wire CLK0_BUF; wire GND_BIT; wire [6:0] GND_BUS_7; wire [15:0] GND_BUS_16;
assign GND_BIT = 0;
assign GND_BUS_7 = 7'b0000000;
assign GND_BUS_16 = 16'b0000000000000000; assign CLKIN_IBUFG_OUT = CLKIN_IBUFG; assign CLK0_OUT = CLKFB_IN;
IBUFG CLKIN_IBUFG_INST (.I(CLKIN_IN), .O(CLKIN_IBUFG)); BUFG CLK0_BUFG_INST (.I(CLK0_BUF), .O(CLKFB_IN));
DCM_ADV DCM_ADV_INST (.CLKFB(CLKFB_IN), .CLKIN(CLKIN_IBUFG), .DADDR(GND_BUS_7[6:0]), .DCLK(GND_BIT), .DEN(GND_BIT), .DI(GND_BUS_16[15:0]), .DWE(GND_BIT), .PSCLK(GND_BIT), .PSEN(GND_BIT), .PSINCDEC(GND_BIT), .RST(GND_BIT), .CLKDV(), .CLKFX(), .CLKFX180(), .CLK0(CLK0_BUF), .CLK2X(),
.CLK2X180(), .CLK90(), .CLK180(), .CLK270(), .DO(), .DRDY(),
.LOCKED(LOCKED_OUT), .PSDONE());
defparam DCM_ADV_INST.CLK_FEEDBACK = \ defparam DCM_ADV_INST.CLKDV_DIVIDE = 2.0; defparam DCM_ADV_INST.CLKFX_DIVIDE = 1; defparam DCM_ADV_INST.CLKFX_MULTIPLY = 4;
defparam DCM_ADV_INST.CLKIN_DIVIDE_BY_2 = \ defparam DCM_ADV_INST.CLKIN_PERIOD = 16.129;
defparam DCM_ADV_INST.CLKOUT_PHASE_SHIFT = \ defparam DCM_ADV_INST.DCM_AUTOCALIBRATION = \
defparam DCM_ADV_INST.DCM_PERFORMANCE_MODE = \ defparam DCM_ADV_INST.DESKEW_ADJUST = \ defparam DCM_ADV_INST.DFS_FREQUENCY_MODE = \ defparam DCM_ADV_INST.DLL_FREQUENCY_MODE = \ defparam DCM_ADV_INST.DUTY_CYCLE_CORRECTION = \ defparam DCM_ADV_INST.FACTORY_JF = 16'hF0F0; defparam DCM_ADV_INST.PHASE_SHIFT = 0;
defparam DCM_ADV_INST.STARTUP_WAIT = \endmodule
==========================选择Internal========================= `timescale 1ns / 1ps
module clk1_test(CLKIN_IN, CLK0_OUT, LOCKED_OUT); input CLKIN_IN; output CLK0_OUT; output LOCKED_OUT;
wire CLKFB_IN; wire CLK0_BUF; wire GND_BIT;
wire [6:0] GND_BUS_7; wire [15:0] GND_BUS_16;
assign GND_BIT = 0;
assign GND_BUS_7 = 7'b0000000;
assign GND_BUS_16 = 16'b0000000000000000; assign CLK0_OUT = CLKFB_IN;
BUFG CLK0_BUFG_INST (.I(CLK0_BUF), .O(CLKFB_IN));
DCM_ADV DCM_ADV_INST (.CLKFB(CLKFB_IN), .CLKIN(CLKIN_IN),
.DADDR(GND_BUS_7[6:0]), .DCLK(GND_BIT), .DEN(GND_BIT), .DI(GND_BUS_16[15:0]), .DWE(GND_BIT), .PSCLK(GND_BIT), .PSEN(GND_BIT), .PSINCDEC(GND_BIT), .RST(GND_BIT), .CLKDV(), .CLKFX(), .CLKFX180(), .CLK0(CLK0_BUF), .CLK2X(), .CLK2X180(), .CLK90(), .CLK180(), .CLK270(), .DO(), .DRDY(),
.LOCKED(LOCKED_OUT), .PSDONE());
defparam DCM_ADV_INST.CLK_FEEDBACK = \ defparam DCM_ADV_INST.CLKDV_DIVIDE = 2.0; defparam DCM_ADV_INST.CLKFX_DIVIDE = 1; defparam DCM_ADV_INST.CLKFX_MULTIPLY = 4;
defparam DCM_ADV_INST.CLKIN_DIVIDE_BY_2 = \ defparam DCM_ADV_INST.CLKIN_PERIOD = 16.129;
defparam DCM_ADV_INST.CLKOUT_PHASE_SHIFT = \ defparam DCM_ADV_INST.DCM_AUTOCALIBRATION = \
defparam DCM_ADV_INST.DCM_PERFORMANCE_MODE = \