// *************************************************************************** // *************************************************************************** // Copyright 2014 - 2017 (c) Analog Devices, Inc. All rights reserved. // // This core is distributed in the hope that it will be useful, but WITHOUT ANY // WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR // A PARTICULAR PURPOSE. // // Redistribution and use of source or resulting binaries, with or without modification // of this file, are permitted under one of the following two license terms: // // 1. The GNU General Public License version 2 as published by the // Free Software Foundation, which can be found in the top level directory of // the repository (LICENSE_GPL2), and at: // // OR // // 2. An ADI specific BSD license as noted in the top level directory, or on-line at: // https://github.com/analogdevicesinc/hdl/blob/master/LICENSE_ADIBSD // This will allow to generate bit files and not release the source code, // as long as it attaches to an ADI device. // // *************************************************************************** // *************************************************************************** module util_axis_fifo ( input m_axis_aclk, input m_axis_aresetn, input m_axis_ready, output m_axis_valid, output [DATA_WIDTH-1:0] m_axis_data, output [ADDRESS_WIDTH:0] m_axis_level, input s_axis_aclk, input s_axis_aresetn, output s_axis_ready, input s_axis_valid, input [DATA_WIDTH-1:0] s_axis_data, output s_axis_empty, output [ADDRESS_WIDTH:0] s_axis_room ); parameter DATA_WIDTH = 64; parameter ASYNC_CLK = 1; parameter ADDRESS_WIDTH = 4; parameter S_AXIS_REGISTERED = 1; generate if (ADDRESS_WIDTH == 0) begin reg [DATA_WIDTH-1:0] cdc_sync_fifo_ram; reg s_axis_waddr = 1'b0; reg m_axis_raddr = 1'b0; wire m_axis_waddr; wire s_axis_raddr; sync_bits #( .NUM_OF_BITS(1), .ASYNC_CLK(ASYNC_CLK) ) i_waddr_sync ( .out_clk(m_axis_aclk), .out_resetn(m_axis_aresetn), .in(s_axis_waddr), .out(m_axis_waddr) ); sync_bits #( .NUM_OF_BITS(1), .ASYNC_CLK(ASYNC_CLK) ) i_raddr_sync ( .out_clk(s_axis_aclk), .out_resetn(s_axis_aresetn), .in(m_axis_raddr), .out(s_axis_raddr) ); assign m_axis_valid = m_axis_raddr != m_axis_waddr; assign m_axis_level = m_axis_valid; assign s_axis_ready = s_axis_raddr == s_axis_waddr; assign s_axis_empty = s_axis_ready; assign s_axis_room = s_axis_ready; always @(posedge s_axis_aclk) begin if (s_axis_ready == 1'b1 && s_axis_valid == 1'b1) cdc_sync_fifo_ram <= s_axis_data; end always @(posedge s_axis_aclk) begin if (s_axis_aresetn == 1'b0) begin s_axis_waddr <= 1'b0; end else begin if (s_axis_ready & s_axis_valid) begin s_axis_waddr <= s_axis_waddr + 1'b1; end end end always @(posedge m_axis_aclk) begin if (m_axis_aresetn == 1'b0) begin m_axis_raddr <= 1'b0; end else begin if (m_axis_valid & m_axis_ready) m_axis_raddr <= m_axis_raddr + 1'b1; end end assign m_axis_data = cdc_sync_fifo_ram; end else begin reg [DATA_WIDTH-1:0] ram[0:2**ADDRESS_WIDTH-1]; wire [ADDRESS_WIDTH-1:0] s_axis_waddr; wire [ADDRESS_WIDTH-1:0] m_axis_raddr; wire _m_axis_ready; wire _m_axis_valid; if (ASYNC_CLK == 1) begin fifo_address_gray_pipelined #( .ADDRESS_WIDTH(ADDRESS_WIDTH) ) i_address_gray ( .m_axis_aclk(m_axis_aclk), .m_axis_aresetn(m_axis_aresetn), .m_axis_ready(_m_axis_ready), .m_axis_valid(_m_axis_valid), .m_axis_raddr(m_axis_raddr), .m_axis_level(m_axis_level), .s_axis_aclk(s_axis_aclk), .s_axis_aresetn(s_axis_aresetn), .s_axis_ready(s_axis_ready), .s_axis_valid(s_axis_valid), .s_axis_empty(s_axis_empty), .s_axis_waddr(s_axis_waddr), .s_axis_room(s_axis_room) ); end else begin fifo_address_sync #( .ADDRESS_WIDTH(ADDRESS_WIDTH) ) i_address_sync ( .clk(m_axis_aclk), .resetn(m_axis_aresetn), .m_axis_ready(_m_axis_ready), .m_axis_valid(_m_axis_valid), .m_axis_raddr(m_axis_raddr), .m_axis_level(m_axis_level), .s_axis_ready(s_axis_ready), .s_axis_valid(s_axis_valid), .s_axis_empty(s_axis_empty), .s_axis_waddr(s_axis_waddr), .s_axis_room(s_axis_room) ); end always @(posedge s_axis_aclk) begin if (s_axis_ready == 1'b1 && s_axis_valid == 1'b1) ram[s_axis_waddr] <= s_axis_data; end if (S_AXIS_REGISTERED == 1) begin reg [DATA_WIDTH-1:0] data; reg valid; always @(posedge m_axis_aclk) begin if (m_axis_aresetn == 1'b0) begin valid <= 1'b0; end else begin if (_m_axis_valid) valid <= 1'b1; else if (m_axis_ready) valid <= 1'b0; end end always @(posedge m_axis_aclk) begin if ((~valid || m_axis_ready) && _m_axis_valid) data <= ram[m_axis_raddr]; end assign _m_axis_ready = ~valid || m_axis_ready; assign m_axis_data = data; assign m_axis_valid = valid; end else begin assign _m_axis_ready = m_axis_ready; assign m_axis_valid = _m_axis_valid; assign m_axis_data = ram[m_axis_raddr]; end end endgenerate endmodule