URL
https://opencores.org/ocsvn/amber/amber/trunk
Subversion Repositories amber
[/] [amber/] [trunk/] [hw/] [vlog/] [system/] [afifo.v] - Rev 12
Go to most recent revision | Compare with Previous | Blame | View Log
////////////////////////////////////////////////////////////////// // // // Generic Asynchronous FIFO // // // // This file is part of the Amber project // // http://www.opencores.org/project,amber // // // // Description // // // // Author(s): // // - Conor Santifort, csantifort.amber@gmail.com // // // ////////////////////////////////////////////////////////////////// // // // Copyright (C) 2010 Authors and OPENCORES.ORG // // // // This source file may be used and distributed without // // restriction provided that this copyright statement is not // // removed from the file and that any derivative work contains // // the original copyright notice and the associated disclaimer. // // // // This source file is free software; you can redistribute it // // and/or modify it under the terms of the GNU Lesser General // // Public License as published by the Free Software Foundation; // // either version 2.1 of the License, or (at your option) any // // later version. // // // // This source 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. See the GNU Lesser General Public License for more // // details. // // // // You should have received a copy of the GNU Lesser General // // Public License along with this source; if not, download it // // from http://www.opencores.org/lgpl.shtml // // // ////////////////////////////////////////////////////////////////// module afifo #( parameter D_WIDTH = 32 ) ( input wr_clk, input rd_clk, input [D_WIDTH-1:0] i_data, output [D_WIDTH-1:0] o_data, input i_push, input i_pop, output o_full, output o_empty ); reg [2:0] wr_pointer = 'd0, rd_pointer = 'd0; reg [2:0] wr_pointer_d1 = 'd0, rd_pointer_d1 = 'd0; reg [2:0] wr_pointer_d2 = 'd0, rd_pointer_d2 = 'd0; wire [2:0] wr_pointer_rd, rd_pointer_wr; reg [D_WIDTH-1:0] data [3:0]; always @( posedge wr_clk ) if ( i_push && !o_full ) begin wr_pointer <= wr_pointer + 1'd1; data[wr_pointer[1:0]] <= i_data; end always @( posedge wr_clk ) begin rd_pointer_d1 <= gray8(rd_pointer); rd_pointer_d2 <= rd_pointer_d1; end always @( posedge rd_clk ) if ( i_pop && !o_empty ) rd_pointer <= rd_pointer + 1'd1; always @( posedge rd_clk ) begin wr_pointer_d1 <= gray8(wr_pointer); wr_pointer_d2 <= wr_pointer_d1; end assign wr_pointer_rd = ungray8(wr_pointer_d2); assign rd_pointer_wr = ungray8(rd_pointer_d2); assign o_data = data[rd_pointer[1:0]]; assign o_full = {~wr_pointer[2], wr_pointer[1:0]} == rd_pointer_wr; assign o_empty = wr_pointer_rd == rd_pointer; function [2:0] gray8; input [2:0] binary; begin case(binary) 3'b000 : gray8 = 3'b000; 3'b001 : gray8 = 3'b001; 3'b010 : gray8 = 3'b011; 3'b011 : gray8 = 3'b010; 3'b100 : gray8 = 3'b110; 3'b101 : gray8 = 3'b111; 3'b110 : gray8 = 3'b101; 3'b111 : gray8 = 3'b100; endcase end endfunction function [2:0] ungray8; input [2:0] gray; begin case(gray) 3'b000 : ungray8 = 3'b000; 3'b001 : ungray8 = 3'b001; 3'b011 : ungray8 = 3'b010; 3'b010 : ungray8 = 3'b011; 3'b110 : ungray8 = 3'b100; 3'b111 : ungray8 = 3'b101; 3'b101 : ungray8 = 3'b110; 3'b100 : ungray8 = 3'b111; endcase end endfunction endmodule
Go to most recent revision | Compare with Previous | Blame | View Log