在Xilinx FPGA中,要實現FIFO的功能時,大部分時候會使用兩種方法:
- FIFO Generator IP核
- XPM_FIFO原語
FIFO Generator IP核的優點是有圖形化界面,配置參數非常直觀;缺點是參數一旦固定,想要更改的化就只能重新generate IP核。
XPM_FIFO原語的優點就是參數配置方便。
對于兩者,還有一個非常重要的區別。!!!大小端不一樣!!!
當din的位寬和dout的位寬非對稱時。舉個栗子:當din的位寬為8bit,dout的位寬為32bit時。
FIFO Generator IP核按大端輸出,即先寫進去的數據放在高8bit
XPM_FIFO原語按小端輸出,即先寫進去的數據放在低8bit
下面為RTL設計文件和測試結果。
//-----------------------------------------------------------------------------
//
// Copyright (c) JoeJoe.
//
// Project : fifo_test
// Module : fifo_test.v
// Parent : None
// Children : None
//
// Description:
//
//
//
//
// Parameters:
//
//
//
//
// Local Parameters:
//
// Notes :
//
// Multicycle and False Paths
// Some exist, embedded within the submodules. See the submodule
// descriptions.
//`timescale 1ns/1psmodule fifo_test (input sclk,input srst_n
);//***************************************************************************
// Parameter definitions
//***************************************************************************//***************************************************************************
// Reg declarations
//***************************************************************************reg [2:0] wait_cnt;reg wr_en;reg [7:0] din;wire rd_en;wire prog_full;wire full;wire empty;wire [31:0] dout;reg xpm_wr_en;reg [7:0] xpm_din;wire xpm_rd_en;wire xpm_prog_full;wire xpm_full;wire xpm_empty;wire [31:0] xpm_dout;//***************************************************************************
// Wire declarations
//***************************************************************************//***************************************************************************
// Code
//***************************************************************************// 等待XPM FIFO not busyalways @(posedge sclk) beginif (srst_n == 1'b0) beginwait_cnt <= 'd0;endelse beginif (wait_cnt == 'd7) beginwait_cnt <= wait_cnt;endelse beginwait_cnt <= wait_cnt + 'd1;endendend// 非滿就寫always @(posedge sclk) beginif (srst_n == 1'b0) beginwr_en <= 1'b0;din <= 'd0;endelse beginif (prog_full == 1'b1) beginwr_en <= 1'b0;din <= 'd0;endelse if (wait_cnt == 'd7) beginwr_en <= 1'b1;din <= din + 'd1;endelse beginwr_en <= 1'b0;din <= 'd0;endendend// 非空就讀assign rd_en = ~empty;// 非滿就寫always @(posedge sclk) beginif (srst_n == 1'b0) beginxpm_wr_en <= 1'b0;xpm_din <= 'd0;endelse beginif (xpm_prog_full == 1'b1) beginxpm_wr_en <= 1'b0;xpm_din <= 'd0;endelse if (wait_cnt == 'd7) beginxpm_wr_en <= 1'b1;xpm_din <= din + 'd1;endelse beginxpm_wr_en <= 1'b0;xpm_din <= 'd0;endendend// 非空就讀assign xpm_rd_en = ~xpm_empty;fifo_generator_0 U_FIFO_GENERATOR_0 (.clk ( sclk ) // input wire clk,.srst ( ~srst_n ) // input wire srst,.din ( din ) // input wire [7 : 0] din,.wr_en ( wr_en ) // input wire wr_en,.rd_en ( rd_en ) // input wire rd_en,.dout ( dout ) // output wire [31 : 0] dout,.full ( full ) // output wire full,.empty ( empty ) // output wire empty,.prog_full ( prog_full ) // output wire prog_full);xpm_fifo_sync #(.DOUT_RESET_VALUE ( "0" ), // String.ECC_MODE ( "no_ecc" ), // String.FIFO_MEMORY_TYPE ( "block" ), // String.FIFO_READ_LATENCY ( 0 ), // DECIMAL.FIFO_WRITE_DEPTH ( 1024 ), // DECIMAL.FULL_RESET_VALUE ( 0 ), // DECIMAL.PROG_EMPTY_THRESH ( 10 ), // DECIMAL.PROG_FULL_THRESH ( 500 ), // DECIMAL.RD_DATA_COUNT_WIDTH( 1 ), // DECIMAL.READ_DATA_WIDTH ( 32 ), // DECIMAL.READ_MODE ( "fwft" ), // String.SIM_ASSERT_CHK ( 0 ), // DECIMAL; 0=disable simulation messages, 1=enable simulation messages.USE_ADV_FEATURES ( "0707" ), // String.WAKEUP_TIME ( 0 ), // DECIMAL.WRITE_DATA_WIDTH ( 8 ), // DECIMAL.WR_DATA_COUNT_WIDTH( 1 ) // DECIMAL)U_XPM_FIFO_SYNC (.almost_empty ( ), // 1-bit output: Almost Empty : When asserted, this signal indicates that// only one more read can be performed before the FIFO goes to empty..almost_full ( ), // 1-bit output: Almost Full: When asserted, this signal indicates that// only one more write can be performed before the FIFO is full..data_valid ( ), // 1-bit output: Read Data Valid: When asserted, this signal indicates// that valid data is available on the output bus (dout )..dbiterr ( ), // 1-bit output: Double Bit Error: Indicates that the ECC decoder detected// a double-bit error and data in the FIFO core is corrupted..dout (xpm_dout ), // READ_DATA_WIDTH-bit output: Read Data: The output data bus is driven// when reading the FIFO..empty (xpm_empty ), // 1-bit output: Empty Flag: When asserted, this signal indicates that the// FIFO is empty. Read requests are ignored when the FIFO is empty,// initiating a read while empty is not destructive to the FIFO..full (xpm_full ), // 1-bit output: Full Flag: When asserted, this signal indicates that the// FIFO is full. Write requests are ignored when the FIFO is full,// initiating a write when the FIFO is full is not destructive to the// contents of the FIFO..overflow ( ), // 1-bit output: Overflow: This signal indicates that a write request// (wren) during the prior clock cycle was rejected, because the FIFO is// full. Overflowing the FIFO is not destructive to the contents of the// FIFO..prog_empty ( ), // 1-bit output: Programmable Empty: This signal is asserted when the// number of words in the FIFO is less than or equal to the programmable// empty threshold value. It is de-asserted when the number of words in// the FIFO exceeds the programmable empty threshold value..prog_full (xpm_prog_full), // 1-bit output: Programmable Full: This signal is asserted when the// number of words in the FIFO is greater than or equal to the// programmable full threshold value. It is de-asserted when the number of// words in the FIFO is less than the programmable full threshold value..rd_data_count ( ), // RD_DATA_COUNT_WIDTH-bit output: Read Data Count: This bus indicates the// number of words read from the FIFO..rd_rst_busy ( ), // 1-bit output: Read Reset Busy: Active-High indicator that the FIFO read// domain is currently in a reset state..sbiterr ( ), // 1-bit output: Single Bit Error: Indicates that the ECC decoder detected// and fixed a single-bit error..underflow ( ), // 1-bit output: Underflow: Indicates that the read request (rd_en) during// the previous clock cycle was rejected because the FIFO is empty. Under// flowing the FIFO is not destructive to the FIFO..wr_ack ( ), // 1-bit output: Write Acknowledge: This signal indicates that a write// request(wr_en) during the prior clock cycle is succeeded..wr_data_count ( ), // WR_DATA_COUNT_WIDTH-bit output: Write Data Count: This bus indicates// the number of words written into the FIFO..wr_rst_busy ( ), // 1-bit output: Write Reset Busy: Active-High indicator that the FIFO// write domain is currently in a reset state..din (xpm_din ), // WRITE_DATA_WIDTH-bit input: Write Data: The input data bus used when// writing the FIFO..injectdbiterr (1'b0 ), // 1-bit input: Double Bit Error Injection: Injects a double bit error if// the ECC feature is used on block RAMs or UltraRAM macros..injectsbiterr (1'b0 ), // 1-bit input: Single Bit Error Injection: Injects a single bit error if// the ECC feature is used on block RAMs or UltraRAM macros..rd_en (xpm_rd_en ), // 1-bit input: Read Enable: If the FIFO is not empty, asserting this// signal causes data(on dout) to be read from the FIFO. Must be held// active-low when rd_rst_busy is active high..rst (~srst_n ), // 1-bit input: Reset: Must be synchronous to wr_clk. The clock(s) can be// unstable at the time of applying reset, but reset must be released only// after the clock(s) is/are stable..sleep (1'b0 ), // 1-bit input: Dynamic power saving- If sleep is High, the memory/fifo// block is in power saving mode..wr_clk (sclk ), // 1-bit input: Write clock: Used for write operation. wr_clk must be a// free running clock..wr_en (xpm_wr_en ) // 1-bit input: Write Enable: If the FIFO is not full, asserting this// signal causes data(on din) to be written to the FIFO Must be held// active-low when rst or wr_rst_busy or rd_rst_busy is active high); endmodule
當din的位寬和dout的位寬非對稱時。舉個栗子:當din的位寬為32bit,dout的位寬為8bit時。
FIFO Generator IP核 高8bit先輸出,低8bit最后輸出
XPM_FIFO原語 低8bit先輸出,高8bit最后輸出
下面為RTL設計文件和測試結果。
//-----------------------------------------------------------------------------
//
// Copyright (c) JoeJoe.
//
// Project : fifo_test
// Module : fifo_test.v
// Parent : None
// Children : None
//
// Description:
//
//
//
//
// Parameters:
//
//
//
//
// Local Parameters:
//
// Notes :
//
// Multicycle and False Paths
// Some exist, embedded within the submodules. See the submodule
// descriptions.
//`timescale 1ns/1psmodule fifo_test (input sclk,input srst_n
);//***************************************************************************
// Parameter definitions
//***************************************************************************//***************************************************************************
// Reg declarations
//***************************************************************************reg [2:0] wait_cnt;reg wr_en;reg [31:0] din;wire rd_en;wire prog_full;wire full;wire empty;wire [7:0] dout;reg xpm_wr_en;reg [31:0] xpm_din;wire xpm_rd_en;wire xpm_prog_full;wire xpm_full;wire xpm_empty;wire [7:0] xpm_dout;//***************************************************************************
// Wire declarations
//***************************************************************************//***************************************************************************
// Code
//***************************************************************************// 等待XPM FIFO not busyalways @(posedge sclk) beginif (srst_n == 1'b0) beginwait_cnt <= 'd0;endelse beginif (wait_cnt == 'd7) beginwait_cnt <= wait_cnt;endelse beginwait_cnt <= wait_cnt + 'd1;endendend// 非滿就寫always @(posedge sclk) beginif (srst_n == 1'b0) beginwr_en <= 1'b0;din <= 'd0;endelse beginif (prog_full == 1'b1) beginwr_en <= 1'b0;din <= din;endelse if (wait_cnt == 'd7) beginwr_en <= 1'b1;din <= din + 'd1;endelse beginwr_en <= 1'b0;din <= 'd0;endendend// 非空就讀assign rd_en = ~empty;// 非滿就寫always @(posedge sclk) beginif (srst_n == 1'b0) beginxpm_wr_en <= 1'b0;xpm_din <= 'd0;endelse beginif (xpm_prog_full == 1'b1) beginxpm_wr_en <= 1'b0;xpm_din <= xpm_din;endelse if (wait_cnt == 'd7) beginxpm_wr_en <= 1'b1;xpm_din <= xpm_din + 'd1;endelse beginxpm_wr_en <= 1'b0;xpm_din <= 'd0;endendend// 非空就讀assign xpm_rd_en = ~xpm_empty;// fifo_generator_0 U_FIFO_GENERATOR_0 (// .clk ( sclk ) // input wire clk// ,.srst ( ~srst_n ) // input wire srst// ,.din ( din ) // input wire [7 : 0] din// ,.wr_en ( wr_en ) // input wire wr_en// ,.rd_en ( rd_en ) // input wire rd_en// ,.dout ( dout ) // output wire [31 : 0] dout// ,.full ( full ) // output wire full// ,.empty ( empty ) // output wire empty// ,.prog_full ( prog_full ) // output wire prog_full// );fifo_generator_1 U_FIFO_GENERATOR_1 (.clk ( sclk ) // input wire clk,.srst ( ~srst_n ) // input wire srst,.din ( din ) // input wire [31 : 0] din,.wr_en ( wr_en ) // input wire wr_en,.rd_en ( rd_en ) // input wire rd_en,.dout ( dout ) // output wire [7 : 0] dout,.full ( full ) // output wire full,.empty ( empty ) // output wire empty,.prog_full ( prog_full ) // output wire prog_full);xpm_fifo_sync #(.DOUT_RESET_VALUE ( "0" ), // String.ECC_MODE ( "no_ecc" ), // String.FIFO_MEMORY_TYPE ( "block" ), // String.FIFO_READ_LATENCY ( 0 ), // DECIMAL.FIFO_WRITE_DEPTH ( 256 ), // DECIMAL.FULL_RESET_VALUE ( 0 ), // DECIMAL.PROG_EMPTY_THRESH ( 10 ), // DECIMAL.PROG_FULL_THRESH ( 125 ), // DECIMAL.RD_DATA_COUNT_WIDTH( 1 ), // DECIMAL.READ_DATA_WIDTH ( 8 ), // DECIMAL.READ_MODE ( "fwft" ), // String.SIM_ASSERT_CHK ( 0 ), // DECIMAL; 0=disable simulation messages, 1=enable simulation messages.USE_ADV_FEATURES ( "0707" ), // String.WAKEUP_TIME ( 0 ), // DECIMAL.WRITE_DATA_WIDTH ( 32 ), // DECIMAL.WR_DATA_COUNT_WIDTH( 1 ) // DECIMAL)U_XPM_FIFO_SYNC (.almost_empty ( ), // 1-bit output: Almost Empty : When asserted, this signal indicates that// only one more read can be performed before the FIFO goes to empty..almost_full ( ), // 1-bit output: Almost Full: When asserted, this signal indicates that// only one more write can be performed before the FIFO is full..data_valid ( ), // 1-bit output: Read Data Valid: When asserted, this signal indicates// that valid data is available on the output bus (dout )..dbiterr ( ), // 1-bit output: Double Bit Error: Indicates that the ECC decoder detected// a double-bit error and data in the FIFO core is corrupted..dout (xpm_dout ), // READ_DATA_WIDTH-bit output: Read Data: The output data bus is driven// when reading the FIFO..empty (xpm_empty ), // 1-bit output: Empty Flag: When asserted, this signal indicates that the// FIFO is empty. Read requests are ignored when the FIFO is empty,// initiating a read while empty is not destructive to the FIFO..full (xpm_full ), // 1-bit output: Full Flag: When asserted, this signal indicates that the// FIFO is full. Write requests are ignored when the FIFO is full,// initiating a write when the FIFO is full is not destructive to the// contents of the FIFO..overflow ( ), // 1-bit output: Overflow: This signal indicates that a write request// (wren) during the prior clock cycle was rejected, because the FIFO is// full. Overflowing the FIFO is not destructive to the contents of the// FIFO..prog_empty ( ), // 1-bit output: Programmable Empty: This signal is asserted when the// number of words in the FIFO is less than or equal to the programmable// empty threshold value. It is de-asserted when the number of words in// the FIFO exceeds the programmable empty threshold value..prog_full (xpm_prog_full), // 1-bit output: Programmable Full: This signal is asserted when the// number of words in the FIFO is greater than or equal to the// programmable full threshold value. It is de-asserted when the number of// words in the FIFO is less than the programmable full threshold value..rd_data_count ( ), // RD_DATA_COUNT_WIDTH-bit output: Read Data Count: This bus indicates the// number of words read from the FIFO..rd_rst_busy ( ), // 1-bit output: Read Reset Busy: Active-High indicator that the FIFO read// domain is currently in a reset state..sbiterr ( ), // 1-bit output: Single Bit Error: Indicates that the ECC decoder detected// and fixed a single-bit error..underflow ( ), // 1-bit output: Underflow: Indicates that the read request (rd_en) during// the previous clock cycle was rejected because the FIFO is empty. Under// flowing the FIFO is not destructive to the FIFO..wr_ack ( ), // 1-bit output: Write Acknowledge: This signal indicates that a write// request(wr_en) during the prior clock cycle is succeeded..wr_data_count ( ), // WR_DATA_COUNT_WIDTH-bit output: Write Data Count: This bus indicates// the number of words written into the FIFO..wr_rst_busy ( ), // 1-bit output: Write Reset Busy: Active-High indicator that the FIFO// write domain is currently in a reset state..din (xpm_din ), // WRITE_DATA_WIDTH-bit input: Write Data: The input data bus used when// writing the FIFO..injectdbiterr (1'b0 ), // 1-bit input: Double Bit Error Injection: Injects a double bit error if// the ECC feature is used on block RAMs or UltraRAM macros..injectsbiterr (1'b0 ), // 1-bit input: Single Bit Error Injection: Injects a single bit error if// the ECC feature is used on block RAMs or UltraRAM macros..rd_en (xpm_rd_en ), // 1-bit input: Read Enable: If the FIFO is not empty, asserting this// signal causes data(on dout) to be read from the FIFO. Must be held// active-low when rd_rst_busy is active high..rst (~srst_n ), // 1-bit input: Reset: Must be synchronous to wr_clk. The clock(s) can be// unstable at the time of applying reset, but reset must be released only// after the clock(s) is/are stable..sleep (1'b0 ), // 1-bit input: Dynamic power saving- If sleep is High, the memory/fifo// block is in power saving mode..wr_clk (sclk ), // 1-bit input: Write clock: Used for write operation. wr_clk must be a// free running clock..wr_en (xpm_wr_en ) // 1-bit input: Write Enable: If the FIFO is not full, asserting this// signal causes data(on din) to be written to the FIFO Must be held// active-low when rst or wr_rst_busy or rd_rst_busy is active high); endmodule
參考文檔如下:《FIFO Generator v13.2 Product Guide》(PG057)
FIFO Generator IP,小位寬寫,大位寬讀,大端。
FIFO Generator IP,大位寬寫,小位寬讀。
疑問:XPM_FIFO為什么不可以設置大小端,以及為什么不和FIFO Generator IP統一???