71 lines
1.7 KiB
Verilog
71 lines
1.7 KiB
Verilog
/*****************************************************************************\
|
|
| Copyright (C) 2021-2022 Luke Wren |
|
|
| SPDX-License-Identifier: Apache-2.0 |
|
|
\*****************************************************************************/
|
|
|
|
// Register file
|
|
// Single write port, dual read port
|
|
|
|
`default_nettype none
|
|
|
|
module hazard3_regfile_1w2r #(
|
|
parameter RESET_REGS = 0, // Unsupported for FAKE_DUALPORT
|
|
parameter N_REGS = 16,
|
|
parameter W_DATA = 32,
|
|
parameter W_ADDR = $clog2(W_DATA) // should be localparam. ISIM...
|
|
) (
|
|
input wire clk,
|
|
input wire rst_n,
|
|
|
|
input wire [W_ADDR-1:0] raddr1,
|
|
output reg [W_DATA-1:0] rdata1,
|
|
|
|
input wire [W_ADDR-1:0] raddr2,
|
|
output reg [W_DATA-1:0] rdata2,
|
|
|
|
input wire [W_ADDR-1:0] waddr,
|
|
input wire [W_DATA-1:0] wdata,
|
|
input wire wen
|
|
);
|
|
|
|
generate
|
|
if (RESET_REGS) begin: real_dualport_reset
|
|
// This will presumably always be implemented with flops
|
|
reg [W_DATA-1:0] mem [0:N_REGS-1];
|
|
|
|
integer i;
|
|
always @ (posedge clk or negedge rst_n) begin
|
|
if (!rst_n) begin
|
|
for (i = 0; i < N_REGS; i = i + 1) begin
|
|
mem[i] <= {W_DATA{1'b0}};
|
|
end
|
|
rdata1 <= {W_DATA{1'b0}};
|
|
rdata2 <= {W_DATA{1'b0}};
|
|
end else begin
|
|
if (wen) begin
|
|
mem[waddr] <= wdata;
|
|
end
|
|
rdata1 <= mem[raddr1];
|
|
rdata2 <= mem[raddr2];
|
|
end
|
|
end
|
|
end else begin: real_dualport_noreset
|
|
// This should be inference-compatible on FPGAs with dual-port BRAMs
|
|
reg [W_DATA-1:0] mem [0:N_REGS-1];
|
|
|
|
always @ (posedge clk) begin
|
|
if (wen) begin
|
|
mem[waddr] <= wdata;
|
|
end
|
|
rdata1 <= mem[raddr1];
|
|
rdata2 <= mem[raddr2];
|
|
end
|
|
end
|
|
endgenerate
|
|
|
|
endmodule
|
|
|
|
`ifndef YOSYS
|
|
`default_nettype wire
|
|
`endif
|