OpenCores
URL https://opencores.org/ocsvn/mpmc8/mpmc8/trunk

Subversion Repositories mpmc8

[/] [mpmc8/] [trunk/] [rtl/] [mpmc8.sv] - Rev 2

Compare with Previous | Blame | View Log

`timescale 1ns / 1ps
// ============================================================================
//        __
//   \\__/ o\    (C) 2015-2022  Robert Finch, Waterloo
//    \  __ /    All rights reserved.
//     \/_//     robfinch<remove>@finitron.ca
//       ||
//
// BSD 3-Clause License
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
//    list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
//    this list of conditions and the following disclaimer in the documentation
//    and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
//    contributors may be used to endorse or promote products derived from
//    this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
//
// 5000 LUTs, 28 BRAM
// ============================================================================
//
//`define RED_SCREEN    1'b1
import mpmc8_pkg::*;

module mpmc8(
rst_i, clk100MHz,
clk0, cyc0, stb0, ack0, we0, sel0, adr0, dati0, dato0,
clk1, cs1, cyc1, stb1, ack1, we1, sel1, adr1, dati1, dato1, sr1, cr1, rb1,
clk2, cyc2, stb2, ack2, we2, sel2, adr2, dati2, dato2,
clk3, cyc3, stb3, ack3, we3, sel3, adr3, dati3, dato3,
clk4, cyc4, stb4, ack4, we4, sel4, adr4, dati4, dato4,
clk5, cyc5, stb5, ack5, sel5, adr5, dato5, spriteno,
clk6, cyc6, stb6, ack6, we6, sel6, adr6, dati6, dato6,
clk7, cs7, cyc7, stb7, ack7, we7, sel7, adr7, dati7, dato7, sr7, cr7, rb7,
mem_ui_rst, mem_ui_clk, calib_complete,
rstn, app_addr, app_cmd, app_en, app_wdf_data, app_wdf_end, app_wdf_mask, app_wdf_wren,
app_rd_data, app_rd_data_valid, app_rd_data_end, app_rdy, app_wdf_rdy,
ch, state
);
input rst_i;
input clk100MHz;

// Channel 0 is reserved for bitmapped graphics display.
//
parameter C0W = 128;    // Channel zero width
parameter STREAM0 = TRUE;
parameter STRIPS0 = 3'd7;
input clk0;
input cyc0;
input stb0;
output ack0;
input [C0W/8-1:0] sel0;
input we0;
input [31:0] adr0;
input [C0W-1:0] dati0;
output reg [C0W-1:0] dato0;
reg [C0W-1:0] dato0n;

// Channel 1 is reserved for cpu1
parameter C1W = 128;
parameter STREAM1 = FALSE;
parameter STRIPS1 = 3'd0;
input clk1;
input cs1;
input cyc1;
input stb1;
output ack1;
input we1;
input [C1W/8-1:0] sel1;
input [31:0] adr1;
input [C1W-1:0] dati1;
output reg [C1W-1:0] dato1;
input sr1;
input cr1;
output rb1;

// Channel 2 is reserved for the ethernet controller
parameter C2W = 32;
parameter STREAM2 = FALSE;
parameter STRIPS2 = 3'd0;
input clk2;
input cyc2;
input stb2;
output ack2;
input we2;
input [C2W/8-1:0] sel2;
input [31:0] adr2;
input [C2W-1:0] dati2;
output reg [C2W-1:0] dato2;

// Channel 3 is reserved for the audio controller
parameter C3W = 16;
parameter STREAM3 = TRUE;
parameter STRIPS3 = 3'd0;
input clk3;
input cyc3;
input stb3;
output ack3;
input we3;
input [C3W/8-1:0] sel3;
input [31:0] adr3;
input [C3W-1:0] dati3;
output reg [C3W-1:0] dato3;

// Channel 4 is reserved for the graphics controller
parameter C4W = 128;
parameter STREAM4 = FALSE;
parameter STRIPS4 = 3'd0;
input clk4;
input cyc4;
input stb4;
output ack4;
input we4;
input [C4W/8-1:0] sel4;
input [31:0] adr4;
input [C4W-1:0] dati4;
output reg [C4W-1:0] dato4;

// Channel 5 is reserved for sprite DMA, which is read-only
parameter C5W = 64;
parameter STREAM5 = TRUE;
parameter STRIPS5 = 3'd3;
input clk5;
input cyc5;
input stb5;
output ack5;
input [C5W/8-1:0] sel5;
input [5:0] spriteno;
input [31:0] adr5;
output reg [C5W-1:0] dato5;

// Channel 6 is reserved for the SD/MMC controller
parameter C6W = 128;
parameter STREAM6 = FALSE;
parameter STRIPS6 = 3'd0;
input clk6;
input cyc6;
input stb6;
output ack6;
input we6;
input [C6W/8-1:0] sel6;
input [31:0] adr6;
input [C6W-1:0] dati6;
output reg [C6W-1:0] dato6;

// Channel 7 is reserved for the cpu
parameter C7W = 16;
parameter C7R = 16;
parameter STREAM7 = FALSE;
parameter STRIPS7 = 3'd3;
input clk7;
input cs7;
input cyc7;
input stb7;
output ack7;
input we7;
input [C7W/8-1:0] sel7;
input [31:0] adr7;
input [C7W-1:0] dati7;
output reg [C7R-1:0] dato7;
input sr7;
input cr7;
output rb7;

// MIG interface signals
input mem_ui_rst;
input mem_ui_clk;
input calib_complete;
output rstn;
output [AMSB:0] app_addr;
output [2:0] app_cmd;
output reg app_en;
output reg [127:0] app_wdf_data;
output [15:0] app_wdf_mask;
output app_wdf_end;
output app_wdf_wren;
input [127:0] app_rd_data;
input app_rd_data_valid;
input app_rd_data_end;
input app_rdy;
input app_wdf_rdy;

// Debugging
output reg [3:0] ch;
output [3:0] state;

integer n;
integer n1;
integer n2;

wire [31:0] app_waddr;
wire [31:0] adr;
wire [127:0] dat128;
reg [15:0] wmask;
reg [127:0] rmw_data;

reg [3:0] nch;
reg [5:0] sreg;
reg rstn;
reg [5:0] nack_to = 6'd0;
reg [5:0] spriteno_r;

wire cs0 = cyc0 && stb0 && adr0[31:29]==3'h0;
wire ics1 = cyc1 & stb1 & cs1;
wire cs2 = cyc2 && stb2 && adr2[31:29]==3'h0;
wire cs3 = cyc3 && stb3 && adr3[31:29]==3'h0;
wire cs4 = cyc4 && stb4 && adr4[31:29]==3'h0;
wire cs5 = cyc5 && stb5 && adr5[31:29]==3'h0;
wire cs6 = cyc6 && stb6 && adr6[31:29]==3'h0;
wire ics7 = cyc7 & stb7 & cs7;

reg acki0,acki1,acki2,acki3,acki4,acki5,acki6,acki7;
wire do_wr;
wire do_wr0;
wire do_wr1;
wire do_wr2;
wire do_wr3;
wire do_wr4;
wire do_wr5;
wire do_wr6;
wire do_wr7;

reg [2:0] num_strips;
wire [2:0] req_strip_cnt;               // count of requested strips
wire [2:0] resp_strip_cnt;              // count of response strips
reg [AMSB:0] app_addr;

reg [15:0] refcnt;                              // refreshing not used, its automnatic
reg refreq;
wire refack;
wire [15:0] tocnt;                                      // memory access timeout counter

reg [3:0] resv_ch [0:NAR-1];
reg [31:0] resv_adr [0:NAR-1];

// Cross clock domain signals
wire cs0xx;
wire we0xx;
wire [C0W/8-1:0] sel0xx;
wire [31:0] adr0xx;
wire [C0W-1:0] dati0xx;

wire cs1xx;
wire we1xx;
wire [15:0] sel1xx;
wire [31:0] adr1xx;
wire [127:0] dati1xx;
wire sr1xx;
wire cr1xx;

wire cs2xx;
wire we2xx;
wire [C2W/8-1:0] sel2xx;
wire [31:0] adr2xx;
wire [C2W-1:0] dati2xx;

wire cs3xx;
wire we3xx;
wire [C3W/8-1:0] sel3xx;
wire [31:0] adr3xx;
wire [C3W-1:0] dati3xx;

wire cs4xx;
wire we4xx;
wire [C4W/8-1:0] sel4xx;
wire [31:0] adr4xx;
wire [C4W-1:0] dati4xx;

wire cs5xx;
wire we5xx;
wire [C5W/8-1:0] sel5xx;
wire [31:0] adr5xx;
wire [C5W-1:0] dati5xx;
wire [5:0] spritenoxx;

wire cs6xx;
wire we6xx;
wire [C6W/8-1:0] sel6xx;
wire [31:0] adr6xx;
wire [C6W-1:0] dati6xx;

wire cs7xx;
wire we7xx;
wire [C7W/8-1:0] sel7xx;
wire [31:0] adr7xx;
wire [C7W-1:0] dati7xx;
wire sr7xx;
wire cr7xx;

reg [7:0] to_cnt;

// Terminate the ack signal as soon as the circuit select goes away.
assign ack0 = acki0 & cs0;
assign ack1 = acki1 & ics1;
assign ack2 = acki2 & cs2;
assign ack3 = acki3 & cs3;
assign ack4 = acki4 & cs4;
assign ack5 = acki5 & cs5;
assign ack6 = acki6 & cs6;
assign ack7 = acki7 & ics7;

wire pre_ack0;
wire pre_ack1;
wire pre_ack2;
wire pre_ack3;
wire pre_ack4;
wire pre_ack5;
wire pre_ack6;
wire pre_ack7;

// Used to transition state to IDLE at end of access
// We dont transition to idle until a negative edge is seen on ack
wire ne_acki0;
wire ne_acki1;
wire ne_acki2;
wire ne_acki3;
wire ne_acki4;
wire ne_acki5;
wire ne_acki6;
wire ne_acki7;
wire pe_acki0;
wire pe_acki1;
wire pe_acki2;
wire pe_acki3;
wire pe_acki4;
wire pe_acki5;
wire pe_acki6;
wire pe_acki7;
edge_det ed_acki0 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki0), .pe(pe_acki0), .ne(ne_acki0), .ee());
edge_det ed_acki1 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki1), .pe(pe_acki1), .ne(ne_acki1), .ee());
edge_det ed_acki2 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki2), .pe(pe_acki2), .ne(ne_acki2), .ee());
edge_det ed_acki3 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki3), .pe(pe_acki3), .ne(ne_acki3), .ee());
edge_det ed_acki4 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki4), .pe(pe_acki4), .ne(ne_acki4), .ee());
edge_det ed_acki5 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki5), .pe(pe_acki5), .ne(ne_acki5), .ee());
edge_det ed_acki6 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki6), .pe(pe_acki6), .ne(ne_acki6), .ee());
edge_det ed_acki7 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(acki7), .pe(pe_acki7), .ne(ne_acki7), .ee());

wire [127:0] ch0_rdat_c, ch0_rdat_s, ch0_rdat;
wire [127:0] ch1_rdat, ch1_rdat_c, ch1_rdat_s;
wire [127:0] ch2_rdat, ch2_rdat_c, ch2_rdat_s;
wire [127:0] ch3_rdat, ch3_rdat_c, ch3_rdat_s;
wire [127:0] ch4_rdat, ch4_rdat_c, ch4_rdat_s;
wire [127:0] ch5_rdat, ch5_rdat_c, ch5_rdat_s;
wire [127:0] ch6_rdat, ch6_rdat_c, ch6_rdat_s;
wire [127:0] ch7_rdat, ch7_rdat_c, ch7_rdat_s;
reg [127:0] ch0_rdatr;
reg [127:0] ch1_rdatr;
reg [127:0] ch2_rdatr;
reg [127:0] ch3_rdatr;
reg [127:0] ch4_rdatr;
reg [127:0] ch5_rdatr;
reg [127:0] ch6_rdatr;
reg [127:0] ch7_rdatr;
wire ch0_hit_c, ch0_hit_s;
wire ch1_hit_c, ch1_hit_s;
wire ch2_hit_c, ch2_hit_s;
wire ch3_hit_c, ch3_hit_s;
wire ch4_hit_c, ch4_hit_s;
wire ch5_hit_c, ch5_hit_s;
wire ch6_hit_c, ch6_hit_s;
wire ch7_hit_c, ch7_hit_s;
wire ch0_hit, ch1_hit, ch2_hit, ch3_hit;
wire ch4_hit, ch5_hit, ch6_hit, ch7_hit;

mpmc8_read_cache rc0 (
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(app_rd_data_valid && !(
                (STREAM0 && ch==4'd0) ||
                (STREAM1 && ch==4'd1) ||
                (STREAM2 && ch==4'd2) ||
                (STREAM3 && ch==4'd3) ||
                (STREAM4 && ch==4'd4) ||
                (STREAM5 && ch==4'd5) ||
                (STREAM6 && ch==4'd6) ||
                (STREAM7 && ch==4'd7)
        )),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(state==WRITE_DATA0),
        .rclk0(STREAM0 ? 1'b0 : clk0),
        .radr0(STREAM0 ? 32'h0 : {adr0xx[31:4],4'h0}),
        .rdat0(ch0_rdat_c),
        .hit0(ch0_hit_c),
        .rclk1(STREAM1 ? 1'b0 : mem_ui_clk),
        .radr1(STREAM1 ? 32'h0 : {adr1xx[31:4],4'h0}),
        .rdat1(ch1_rdat_c),
        .hit1(ch1_hit_c),
        .rclk2(STREAM2 ? 1'b0 : mem_ui_clk),
        .radr2(STREAM2 ? 32'h0 : {adr2xx[31:4],4'h0}),
        .rdat2(ch2_rdat_c),
        .hit2(ch2_hit_c),
        .rclk3(STREAM3 ? 1'b0 : mem_ui_clk),
        .radr3(STREAM3 ? 32'h0 : {adr3xx[31:4],4'h0}),
        .rdat3(ch3_rdat_c),
        .hit3(ch3_hit_c),
        .rclk4(STREAM4 ? 1'b0 : mem_ui_clk),
        .radr4(STREAM4 ? 32'h0 : {adr4xx[31:4],4'h0}),
        .rdat4(ch4_rdat_c),
        .hit4(ch4_hit_c),
        .rclk5(STREAM5 ? 1'b0 : mem_ui_clk),
        .radr5(STREAM5 ? 32'h0 : {adr5xx[31:4],4'h0}),
        .rdat5(ch5_rdat_c),
        .hit5(ch5_hit_c),
        .rclk6(STREAM6 ? 1'b0 : mem_ui_clk),
        .radr6(STREAM6 ? 32'h0 : {adr6xx[31:4],4'h0}),
        .rdat6(ch6_rdat_c),
        .hit6(ch6_hit_c),
        .rclk7(STREAM7 ? 1'b0 : mem_ui_clk),
        .radr7(STREAM7 ? 32'h0 : {adr7xx[31:4],4'h0}),
        .rdat7(ch7_rdat_c),
        .hit7(ch7_hit_c)
);

mpmc8_strm_read_cache ustrm0
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd0 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr0xx[31:4],4'h0}),
        .rdat(ch0_rdat_s),
        .hit(ch0_hit_s)
);

mpmc8_strm_read_cache ustrm1
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd1 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr1xx[31:4],4'h0}),
        .rdat(ch1_rdat_s),
        .hit(ch1_hit_s)
);

mpmc8_strm_read_cache ustrm2
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd2 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr2xx[31:4],4'h0}),
        .rdat(ch2_rdat_s),
        .hit(ch2_hit_s)
);

mpmc8_strm_read_cache ustrm3
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd3 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr3xx[31:4],4'h0}),
        .rdat(ch3_rdat_s),
        .hit(ch3_hit_s)
);

mpmc8_strm_read_cache ustrm4
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd4 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr4xx[31:4],4'h0}),
        .rdat(ch4_rdat_s),
        .hit(ch4_hit_s)
);

mpmc8_spr_read_cache usprrc5
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .spr_num(spriteno[4:0]),
        .wr(ch==4'd5 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr5xx[31:4],4'h0}),
        .rdat(ch5_rdat_s),
        .hit(ch5_hit_s)
);

mpmc8_strm_read_cache ustrm6
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd6 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr6xx[31:4],4'h0}),
        .rdat(ch6_rdat_s),
        .hit(ch6_hit_s)
);

mpmc8_strm_read_cache ustrm7
(
        .rst(rst_i),
        .wclk(mem_ui_clk),
        .wr(ch==4'd7 && app_rd_data_valid),
        .wadr({app_waddr[31:4],4'h0}),
        .wdat(app_rd_data),
        .inv(1'b0),
        .rclk(mem_ui_clk),
        .radr({adr7xx[31:4],4'h0}),
        .rdat(ch7_rdat_s),
        .hit(ch7_hit_s)
);


assign ch0_hit = STREAM0 ? ch0_hit_s : ch0_hit_c;
assign ch0_rdat = STREAM0 ? ch0_rdat_s : ch0_rdat_c;
assign ch1_hit = STREAM1 ? ch1_hit_s : ch1_hit_c;
assign ch1_rdat = STREAM1 ? ch1_rdat_s : ch1_rdat_c;
assign ch2_hit = STREAM2 ? ch2_hit_s : ch2_hit_c;
assign ch2_rdat = STREAM2 ? ch2_rdat_s : ch2_rdat_c;
assign ch3_hit = STREAM3 ? ch3_hit_s : ch3_hit_c;
assign ch3_rdat = STREAM3 ? ch3_rdat_s : ch3_rdat_c;
assign ch4_hit = STREAM4 ? ch4_hit_s : ch4_hit_c;
assign ch4_rdat = STREAM4 ? ch4_rdat_s : ch4_rdat_c;
assign ch5_hit = STREAM5 ? ch5_hit_s : ch5_hit_c;
assign ch5_rdat = STREAM5 ? ch5_rdat_s : ch5_rdat_c;
assign ch6_hit = STREAM6 ? ch6_hit_s : ch6_hit_c;
assign ch6_rdat = STREAM6 ? ch6_rdat_s : ch6_rdat_c;
assign ch7_hit = STREAM7 ? ch7_hit_s : ch7_hit_c;
assign ch7_rdat = STREAM7 ? ch7_rdat_s : ch7_rdat_c;

// Register signals onto mem_ui_clk domain
mpmc8_sync #(.W(C0W)) usyn0
(
        .clk(mem_ui_clk),
        .cs_i(cs0),
        .we_i(we0),
        .sel_i(sel0),
        .adr_i(adr0),
        .dati_i(dati0),
        .sr_i(1'b0),
        .cr_i(1'b0),
        .cs_o(cs0xx),
        .we_o(we0xx),
        .sel_o(sel0xx),
        .adr_o(adr0xx),
        .dati_o(dati0xx),
        .sr_o(),
        .cr_o()
);

mpmc8_sync #(.W(C1W)) usyn1
(
        .clk(mem_ui_clk),
        .cs_i(ics1),
        .we_i(we1),
        .sel_i(sel1),
        .adr_i(adr1),
        .dati_i(dati1),
        .sr_i(sr1),
        .cr_i(cr1),
        .cs_o(cs1xx),
        .we_o(we1xx),
        .sel_o(sel1xx),
        .adr_o(adr1xx),
        .dati_o(dati1xx),
        .sr_o(sr1xx),
        .cr_o(cr1xx)
);

mpmc8_sync #(.W(C2W)) usyn2
(
        .clk(mem_ui_clk),
        .cs_i(cs2),
        .we_i(we2),
        .sel_i(sel2),
        .adr_i(adr2),
        .dati_i(dati2),
        .sr_i(1'b0),
        .cr_i(1'b0),
        .cs_o(cs2xx),
        .we_o(we2xx),
        .sel_o(sel2xx),
        .adr_o(adr2xx),
        .dati_o(dati2xx),
        .sr_o(),
        .cr_o()
);

mpmc8_sync #(.W(C3W)) usyn3
(
        .clk(mem_ui_clk),
        .cs_i(cs3),
        .we_i(we3),
        .sel_i(sel3),
        .adr_i(adr3),
        .dati_i(dati3),
        .sr_i(1'b0),
        .cr_i(1'b0),
        .cs_o(cs3xx),
        .we_o(we3xx),
        .sel_o(sel3xx),
        .adr_o(adr3xx),
        .dati_o(dati3xx),
        .sr_o(),
        .cr_o()
);

mpmc8_sync #(.W(C4W)) usyn4
(
        .clk(mem_ui_clk),
        .cs_i(cs4),
        .we_i(we4),
        .sel_i(sel4),
        .adr_i(adr4),
        .dati_i(dati4),
        .sr_i(1'b0),
        .cr_i(1'b0),
        .cs_o(cs4xx),
        .we_o(we4xx),
        .sel_o(sel4xx),
        .adr_o(adr4xx),
        .dati_o(dati4xx),
        .sr_o(),
        .cr_o()
);

mpmc8_sync #(.W(C5W)) usyn5
(
        .clk(mem_ui_clk),
        .cs_i(cs5),
        .we_i(1'b0),
        .sel_i(sel5),
        .adr_i(adr5),
        .dati_i(dati5),
        .sr_i(1'b0),
        .cr_i(1'b0),
        .cs_o(cs5xx),
        .we_o(),
        .sel_o(sel5xx),
        .adr_o(adr5xx),
        .dati_o(dati5xx),
        .sr_o(),
        .cr_o()
);

mpmc8_sync #(.W(C6W)) usyn6
(
        .clk(mem_ui_clk),
        .cs_i(cs6),
        .we_i(we6),
        .sel_i(sel6),
        .adr_i(adr6),
        .dati_i(dati6),
        .sr_i(1'b0),
        .cr_i(1'b0),
        .cs_o(cs6xx),
        .we_o(we6xx),
        .sel_o(sel6xx),
        .adr_o(adr6xx),
        .dati_o(dati6xx),
        .sr_o(),
        .cr_o()
);

mpmc8_sync #(.W(C7R)) usyn7
(
        .clk(mem_ui_clk),
        .cs_i(cs7),
        .we_i(we7),
        .sel_i(sel7),
        .adr_i(adr7),
        .dati_i(dati7),
        .sr_i(sr7),
        .cr_i(cr7),
        .cs_o(cs7xx),
        .we_o(we7xx),
        .sel_o(sel7xx),
        .adr_o(adr7xx),
        .dati_o(dati7xx),
        .sr_o(sr7xx),
        .cr_o(cr7xx)
);

reg [19:0] rst_ctr;
always @(posedge clk100MHz)
if (rst_i)
        rst_ctr <= 24'd0;
else begin
        if (!rst_ctr[15])
                rst_ctr <= rst_ctr + 2'd1;
        rstn <= rst_ctr[15];
end

reg toggle;     // CPU1 / CPU0 priority toggle
reg toggle_sr;
reg [19:0] resv_to_cnt;
reg sr1x,sr7x;
reg [127:0] dati128;

// Detect cache hits on read caches
// For the sprite channel, reading the 64-bits of a strip not beginning at
// a 64-byte aligned paragraph only checks for the 64 bit address adr[5:3].
// It's assumed that a 4x128-bit strips were read the by the previous access.
// It's also assumed that the strip address won't match because there's more
// than one sprite and sprite accesses are essentially random.
wire ch0_taghit = cs0xx && !we0xx && ch0_hit;
wire ch1_taghit = ics1 && !we1xx && cs1xx && ch1_hit;
wire ch2_taghit = !we2xx && cs2xx && ch2_hit;
wire ch3_taghit = !we3xx && cs3xx && ch3_hit;
wire ch4_taghit = !we4xx && cs4xx && ch4_hit;
wire ch5_taghit = cs5xx && ch5_hit;
wire ch6_taghit = !we6xx && cs6xx && ch6_hit;
wire ch7_taghit = cs7xx && !we7xx && ch7_hit;

always_comb
begin
        sr1x = FALSE;
  if (ch1_taghit)
    sr1x = sr1xx;
end
always_comb
begin
        sr7x = FALSE;
  if (ch7_taghit)
    sr7x = sr7xx;
end

// Select the channel
mpmc8_ch_prioritize uchp1
(
        .clk(mem_ui_clk),
        .state(state),
        .cs0(cs0xx),
        .cs1(cs1xx),
        .cs2(cs2xx),
        .cs3(cs3xx),
        .cs4(cs4xx),
        .cs5(cs5xx),
        .cs6(cs6xx),
        .cs7(cs7xx),
        .we0(we0xx),
        .we1(we1xx),
        .we2(we2xx),
        .we3(we3xx),
        .we4(we4xx),
        .we5(1'b0),
        .we6(we6xx),
        .we7(we7xx),
        .ch0_taghit(ch0_taghit),
        .ch1_taghit(ch1_taghit),
        .ch2_taghit(ch2_taghit),
        .ch3_taghit(ch3_taghit),
        .ch4_taghit(ch4_taghit),
        .ch5_taghit(ch5_taghit),
        .ch6_taghit(ch6_taghit),
        .ch7_taghit(ch7_taghit),
        .ch(nch)
);


always_ff @(posedge mem_ui_clk)
if (state==IDLE || (state==PRESET1 && do_wr))
        ch <= nch;

mpmc8_addr_select 
#(
        .S0(STRIPS0),
        .S1(STRIPS1),
        .S2(STRIPS2),
        .S3(STRIPS3),
        .S4(STRIPS4),
        .S5(STRIPS5),
        .S6(STRIPS6),
        .S7(STRIPS7)
)
uadrs1
(
        .rst(mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(nch),
        .we0(we0xx),
        .we1(we1xx),
        .we2(we2xx),
        .we3(we3xx),
        .we4(we4xx),
        .we5(1'b0),
        .we6(we6xx),
        .we7(we7xx),
        .adr0(adr0xx),
        .adr1(adr1xx),
        .adr2(adr2xx),
        .adr3(adr3xx),
        .adr4(adr4xx),
        .adr5(adr5xx),
        .adr6(adr6xx),
        .adr7(adr7xx),
        .adr(adr)
);

wire [2:0] app_addr3;

mpmc8_addr_gen uagen1
(
        .rst(mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .rdy(app_rdy),
        .num_strips(num_strips),
        .strip_cnt(req_strip_cnt),
        .addr_base(adr),
        .addr({app_addr3,app_addr})
);

mpmc8_waddr_gen uwadgen1
(
        .rst(mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .valid(app_rd_data_valid),
        .num_strips(num_strips),
        .strip_cnt(resp_strip_cnt),
        .addr_base(adr),
        .addr(app_waddr)
);

// Setting the write mask
reg [15:0] wmask0;
reg [15:0] wmask1;
reg [15:0] wmask2;
reg [15:0] wmask3;
reg [15:0] wmask4;
reg [15:0] wmask5;
reg [15:0] wmask6;
reg [15:0] wmask7;

mpmc8_set_write_mask
#(
        .C0W(C0W),
        .C1W(C1W),
        .C2W(C2W),
        .C3W(C3W),
        .C4W(C4W),
        .C5W(C5W),
        .C6W(C6W),
        .C7W(C7W)
)
uswm1
(
        .clk(mem_ui_clk),
        .state(state),
        .we0(we0xx),
        .we1(we1xx),
        .we2(we2xx),
        .we3(we3xx),
        .we4(we4xx),
        .we5(1'b0),
        .we6(we6xx),
        .we7(we7xx),
        .sel0(sel0xx),
        .sel1(sel1xx),
        .sel2(sel2xx),
        .sel3(sel3xx),
        .sel4(sel4xx),
        .sel5(sel5xx),
        .sel6(sel6xx),
        .sel7(sel7xx),
        .adr0(adr0xx),
        .adr1(adr1xx),
        .adr2(adr2xx),
        .adr3(adr3xx),
        .adr4(adr4xx),
        .adr5(adr5xx),
        .adr6(adr6xx),
        .adr7(adr7xx),
        .mask0(wmask0),
        .mask1(wmask1),
        .mask2(wmask2),
        .mask3(wmask3),
        .mask4(wmask4),
        .mask5(wmask5),
        .mask6(wmask6),
        .mask7(wmask7)
);

// Setting the write data
mpmc8_data_select
#(
        .C0W(C0W),
        .C1W(C1W),
        .C2W(C2W),
        .C3W(C3W),
        .C4W(C4W),
        .C5W(C5W),
        .C6W(C6W),
        .C7W(C7W)
)
uds1
(
        .clk(mem_ui_clk),
        .state(state),
        .ch(nch),
        .dati0(dati0xx),
        .dati1(dati1xx),
        .dati2(dati2xx),
        .dati3(dati3xx),
        .dati4(dati4xx),
        .dati5('d0),
        .dati6(dati6xx),
        .dati7(dati7xx),
        .dato(dat128)
);

// Setting the data value. Unlike reads there is only a single strip involved.
// Force unselected byte lanes to $FF
wire [15:0] mem_wdf_mask2;
reg [127:0] dat128x;
genvar g;
generate begin : gMemData
        for (g = 0; g < 16; g = g + 1)
                always_comb
                        if (mem_wdf_mask2[g])
                                dat128x[g*8+7:g*8] = 8'hFF;
                        else
                                dat128x[g*8+7:g*8] = dat128[g*8+7:g*8];
end
endgenerate

always_ff @(posedge mem_ui_clk)
if (mem_ui_rst)
  app_wdf_data <= 128'd0;
else begin
        if (state==PRESET2)
                app_wdf_data <= dat128x;
        else if (state==WRITE_TRAMP1)
                app_wdf_data <= rmw_data;
end

generate begin : gRMW
for (g = 0; g < 16; g = g + 1) begin
        always_ff @(posedge mem_ui_clk)
        begin
                if (state==WRITE_TRAMP) begin
                        if (app_wdf_mask[g])
                                rmw_data[g*8+7:g*8] <= app_wdf_data[g*8+7:g*8];
                        else
                                rmw_data[g*8+7:g*8] <= app_rd_data[g*8+7:g*8];
                end
        end
end
end
endgenerate

always_ff @(posedge mem_ui_clk)
if (state==IDLE)
        spriteno_r <= spritenoxx;

// Setting burst length
mpmc8_set_num_strips 
#(
        .S0(STRIPS0),
        .S1(STRIPS1),
        .S2(STRIPS2),
        .S3(STRIPS3),
        .S4(STRIPS4),
        .S5(STRIPS5),
        .S6(STRIPS6),
        .S7(STRIPS7)
)
usns1
(
        .clk(mem_ui_clk),
        .state(state),
        .ch(nch),
        .we0(we0xx),
        .we1(we1xx),
        .we2(we2xx),
        .we3(we3xx),
        .we4(we4xx),
        .we5(we5xx),
        .we6(we6xx),
        .we7(we7xx),
        .num_strips(num_strips)
);

// Setting the data mask. Values are enabled when the data mask is zero.
mpmc8_mask_select umsks1
(
        .rst(mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .wmask0(wmask0),
        .wmask1(wmask1),
        .wmask2(wmask2),
        .wmask3(wmask3),
        .wmask4(wmask4),
        .wmask5(wmask5),
        .wmask6(wmask6),
        .wmask7(wmask7),
        .mask(app_wdf_mask),
        .mask2(mem_wdf_mask2)
);

// Setting output data. Force output data to zero when not selected to allow
// wire-oring the data.
mpmc8_data_output udo1
(
        .clk0(clk0), .cs0(cs0), .adr0(adr0), .ch0_rdat(ch0_rdat), .dato0(dato0),
        .clk1(clk1), .cs1(cs1), .adr1(adr1), .ch1_rdat(ch1_rdat), .dato1(dato1),
        .clk2(clk2), .cs2(cs2), .adr2(adr2), .ch2_rdat(ch2_rdat), .dato2(dato2),
        .clk3(clk3), .cs3(cs3), .adr3(adr3), .ch3_rdat(ch3_rdat), .dato3(dato3),
        .clk4(clk4), .cs4(cs4), .adr4(adr4), .ch4_rdat(ch4_rdat), .dato4(dato4),
        .clk5(clk5), .cs5(cs5), .adr5(adr5), .ch5_rdat(ch5_rdat), .dato5(dato5),
        .clk6(clk6), .cs6(cs6), .adr6(adr6), .ch6_rdat(ch6_rdat), .dato6(dato6),
        .clk7(clk7), .cs7(cs7), .adr7(adr7), .ch7_rdat(ch7_rdat), .dato7(dato7)
);

mpmc8_ack_gen #(.N(0)) uackg0
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs0),
        .adr(adr0xx),
        .cr(1'b0),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch0_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki0),
        .pre_ack(pre_ack0)
);

mpmc8_ack_gen #(.N(1)) uackg1
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs1&ics1),
        .adr(adr1xx),
        .cr(cr1xx),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch1_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki1),
        .pre_ack(pre_ack1)
);

mpmc8_ack_gen #(.N(2)) uackg2
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs2),
        .adr(adr2xx),
        .cr(1'b0),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch2_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki2),
        .pre_ack(pre_ack2)
);

mpmc8_ack_gen #(.N(3)) uackg3
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs3),
        .adr(adr3xx),
        .cr(1'b0),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch3_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki3),
        .pre_ack(pre_ack3)
);

mpmc8_ack_gen #(.N(4)) uackg4
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs4),
        .adr(adr4xx),
        .cr(1'b0),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch4_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki4),
        .pre_ack(pre_ack4)
);

mpmc8_ack_gen #(.N(5)) uackg5
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs5),
        .adr(adr5xx),
        .cr(1'b0),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch5_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki5),
        .pre_ack(pre_ack5)
);

mpmc8_ack_gen #(.N(6)) uackg6
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(cs6),
        .adr(adr6xx),
        .cr(1'b0),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch6_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki6),
        .pre_ack(pre_ack6)
);

mpmc8_ack_gen #(.N(7)) uackg7
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .ch(ch),
        .cs(ics7),
        .adr(adr7xx),
        .cr(cr7xx),
        .wr(do_wr),
        .to(tocnt[9]),
        .taghit(ch7_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .ack(acki7),
        .pre_ack(pre_ack7)
);

wire ne_data_valid;
edge_det ued1 (.rst(rst_i), .clk(mem_ui_clk), .ce(1'b1), .i(app_rd_data_valid), .pe(), .ne(ne_data_valid), .ee());

mpmc8_state_machine ustmac1
(
        .rst(rst_i|mem_ui_rst),
        .clk(mem_ui_clk),
        .ch(nch), 
        .acki0(acki0),
        .acki1(acki1),
        .acki2(acki2),
        .acki3(acki3),
        .acki4(acki4),
        .acki5(acki5),
        .acki6(acki6),
        .acki7(acki7),
        .ch0_taghit(ch0_taghit),
        .ch1_taghit(ch1_taghit),
        .ch2_taghit(ch2_taghit),
        .ch3_taghit(ch3_taghit),
        .ch4_taghit(ch4_taghit),
        .ch5_taghit(ch5_taghit),
        .ch6_taghit(ch6_taghit),
        .ch7_taghit(ch7_taghit),
        .wdf_rdy(app_wdf_rdy),
        .rdy(app_rdy),
        .do_wr(do_wr),
        .rd_data_valid(app_rd_data_valid),
        .num_strips(num_strips),
        .req_strip_cnt(req_strip_cnt),
        .resp_strip_cnt(resp_strip_cnt),
        .to(tocnt[9]),
        .cr1(cr1xx),
        .cr7(cr7xx),
        .adr1(adr1xx),
        .adr7(adr7xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .state(state)
);


wire [3:0] prev_state;

mpmc8_to_cnt utoc1
(
        .clk(mem_ui_clk),
        .state(state),
        .prev_state(prev_state),
        .to_cnt(tocnt)
);

mpmc8_prev_state upst1
(
        .clk(mem_ui_clk),
        .state(state),
        .prev_state(prev_state)
);

mpmc8_app_en_gen ueng1
(
        .clk(mem_ui_clk),
        .state(state),
        .rdy(app_rdy),
        .strip_cnt(req_strip_cnt),
        .num_strips(num_strips),
        .en(app_en)
);

mpmc8_app_cmd_gen ucg1
(
        .clk(mem_ui_clk),
        .state(state),
        .cmd(app_cmd)
);

mpmc8_app_wdf_wren_gen uwreng1
(
        .clk(mem_ui_clk),
        .state(state),
        .rdy(app_wdf_rdy),
        .wren(app_wdf_wren)
);

mpmc8_app_wdf_end_gen uwendg1
(
        .clk(mem_ui_clk),
        .state(state),
        .rdy(app_wdf_rdy),
        .wend(app_wdf_end)
);

mpmc8_req_strip_cnt ursc1
(
        .clk(mem_ui_clk),
        .state(state),
        .wdf_rdy(app_wdf_rdy),
        .rdy(app_rdy),
        .num_strips(num_strips),
        .strip_cnt(req_strip_cnt)
);

mpmc8_resp_strip_cnt urespsc1
(
        .clk(mem_ui_clk),
        .state(state),
        .valid(app_rd_data_valid),
        .num_strips(num_strips),
        .strip_cnt(resp_strip_cnt)
);


mpmc8_do_wr udowr0
(
        .we(we0xx),
        .cr(1'b0),
        .adr(adr0xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr0)
);

mpmc8_do_wr udowr1
(
        .we(we1xx),
        .cr(cr1xx),
        .adr(adr1xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr1)
);

mpmc8_do_wr udowr2
(
        .we(we2xx),
        .cr(1'b0),
        .adr(adr2xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr2)
);

mpmc8_do_wr udowr3
(
        .we(we3xx),
        .cr(1'b0),
        .adr(adr3xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr3)
);

mpmc8_do_wr udowr4
(
        .we(we4xx),
        .cr(1'b0),
        .adr(adr4xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr4)
);

mpmc8_do_wr udowr5
(
        .we(1'b0),
        .cr(1'b0),
        .adr(adr5xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr5)
);

mpmc8_do_wr udowr6
(
        .we(we6xx),
        .cr(1'b0),
        .adr(adr6xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr6)
);

mpmc8_do_wr udowr7
(
        .we(we7xx),
        .cr(cr7xx),
        .adr(adr7xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .do_wr(do_wr7)
);

mpmc8_do_wr_select udws1
(
        .clk(mem_ui_clk),
        .state(state),
        .ch(nch),
        .wr0(do_wr0),
        .wr1(do_wr1),
        .wr2(do_wr2),
        .wr3(do_wr3),
        .wr4(do_wr4),
        .wr5(do_wr5),
        .wr6(do_wr6),
        .wr7(do_wr7),
        .wr(do_wr)
);

// Reservation status bit
mpmc8_resv_bit ursb1
(
        .clk(mem_ui_clk),
        .state(state),
        .cs(cs1xx),
        .we(we1xx),
        .ack(acki1),
        .cr(cr1xx),
        .adr(adr1xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .rb(rb1)
);

mpmc8_resv_bit ursb7
(
        .clk(mem_ui_clk),
        .state(state),
        .cs(cs7xx),
        .we(we7xx),
        .ack(acki7),
        .cr(cr7xx),
        .adr(adr7xx),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr),
        .rb(rb7)
);

// Managing address reservations
mpmc8_addr_resv_man uarman1
(
        .rst(mem_ui_rst),
        .clk(mem_ui_clk),
        .state(state),
        .cs1(cs1xx),
        .ack1(acki1),
        .we1(we1xx),
        .adr1(adr1xx),
        .sr1(sr1x),
        .cr1(cr1xx),
        .ch1_taghit(ch1_taghit),
        .cs7(cs7xx),
        .ack7(acki7),
        .we7(we7xx),
        .adr7(adr7xx),
        .sr7(sr7x),
        .cr7(cr7xx),
        .ch7_taghit(ch7_taghit),
        .resv_ch(resv_ch),
        .resv_adr(resv_adr)
);

endmodule

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.