URL
https://opencores.org/ocsvn/amber/amber/trunk
Subversion Repositories amber
[/] [amber/] [trunk/] [hw/] [vlog/] [amber25/] [a25_wishbone_buf.v] - Rev 57
Go to most recent revision | Compare with Previous | Blame | View Log
////////////////////////////////////////////////////////////////// // // // Wishbone master interface port buffer // // // // This file is part of the Amber project // // http://www.opencores.org/project,amber // // // // Description // // This is a sub-module of the Amber wishbone master // // interface. The wishbone master interface connects a number // // of internal amber ports to the wishbone bus. The ports // // are; // // instruction cache read accesses // // data cache read and write accesses (cached) // // data cache read and write accesses (uncached) // // // // The buffer module buffers a single port. For write // // requests, this allows the processor core to continue // // executing withont having to wait for the wishbone write // // operation to complete. // // // // Author(s): // // - Conor Santifort, csantifort.amber@gmail.com // // // ////////////////////////////////////////////////////////////////// // // // Copyright (C) 2011 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 a25_wishbone_buf ( input i_clk, // Core side input i_req, input i_write, input [127:0] i_wdata, input [15:0] i_be, input [31:0] i_addr, output [127:0] o_rdata, output o_ack, // Wishbone side output o_valid, input i_accepted, output o_write, output [127:0] o_wdata, output [15:0] o_be, output [31:0] o_addr, input [127:0] i_rdata, input i_rdata_valid ); // ---------------------------------------------------- // Signals // ---------------------------------------------------- reg [1:0] wbuf_used_r = 'd0; reg [127:0] wbuf_wdata_r [1:0]; reg [31:0] wbuf_addr_r [1:0]; reg [15:0] wbuf_be_r [1:0]; reg [1:0] wbuf_write_r = 'd0; reg wbuf_wp_r = 'd0; // write buf write pointer reg wbuf_rp_r = 'd0; // write buf read pointer reg busy_reading_r = 'd0; reg wait_rdata_valid_r = 'd0; wire in_wreq; reg ack_owed_r = 'd0; // ---------------------------------------------------- // Access Buffer // ---------------------------------------------------- assign in_wreq = i_req && i_write; assign push = i_req && !busy_reading_r && (wbuf_used_r == 2'd1 || (wbuf_used_r == 2'd0 && !i_accepted)); assign pop = o_valid && i_accepted && wbuf_used_r != 2'd0; always @(posedge i_clk) if (push && pop) wbuf_used_r <= wbuf_used_r; else if (push) wbuf_used_r <= wbuf_used_r + 1'd1; else if (pop) wbuf_used_r <= wbuf_used_r - 1'd1; always @(posedge i_clk) if (push && in_wreq && !o_ack) ack_owed_r = 1'd1; else if (!i_req && o_ack) ack_owed_r = 1'd0; always @(posedge i_clk) if (push) begin wbuf_wdata_r [wbuf_wp_r] <= i_wdata; wbuf_addr_r [wbuf_wp_r] <= i_addr; wbuf_be_r [wbuf_wp_r] <= i_write ? i_be : 16'hffff; wbuf_write_r [wbuf_wp_r] <= i_write; wbuf_wp_r <= !wbuf_wp_r; end always @(posedge i_clk) if (pop) wbuf_rp_r <= !wbuf_rp_r; // ---------------------------------------------------- // Output logic // ---------------------------------------------------- assign o_wdata = wbuf_used_r != 2'd0 ? wbuf_wdata_r[wbuf_rp_r] : i_wdata; assign o_write = wbuf_used_r != 2'd0 ? wbuf_write_r[wbuf_rp_r] : i_write; assign o_addr = wbuf_used_r != 2'd0 ? wbuf_addr_r [wbuf_rp_r] : i_addr; assign o_be = wbuf_used_r != 2'd0 ? wbuf_be_r [wbuf_rp_r] : i_write ? i_be : 16'hffff; assign o_ack = (in_wreq ? (wbuf_used_r == 2'd0) : i_rdata_valid) || (ack_owed_r && pop); assign o_valid = (wbuf_used_r != 2'd0 || i_req) && !wait_rdata_valid_r; assign o_rdata = i_rdata; always@(posedge i_clk) if (o_valid && !o_write) busy_reading_r <= 1'd1; else if (i_rdata_valid) busy_reading_r <= 1'd0; always@(posedge i_clk) if (o_valid && !o_write && i_accepted) wait_rdata_valid_r <= 1'd1; else if (i_rdata_valid) wait_rdata_valid_r <= 1'd0; endmodule
Go to most recent revision | Compare with Previous | Blame | View Log