[6] | 1 | // ========== Copyright Header Begin ========================================== |
---|
| 2 | // |
---|
| 3 | // OpenSPARC T1 Processor File: sparc_tlu_intdp.v |
---|
| 4 | // Copyright (c) 2006 Sun Microsystems, Inc. All Rights Reserved. |
---|
| 5 | // DO NOT ALTER OR REMOVE COPYRIGHT NOTICES. |
---|
| 6 | // |
---|
| 7 | // The above named program is free software; you can redistribute it and/or |
---|
| 8 | // modify it under the terms of the GNU General Public |
---|
| 9 | // License version 2 as published by the Free Software Foundation. |
---|
| 10 | // |
---|
| 11 | // The above named program is distributed in the hope that it will be |
---|
| 12 | // useful, but WITHOUT ANY WARRANTY; without even the implied warranty of |
---|
| 13 | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
---|
| 14 | // General Public License for more details. |
---|
| 15 | // |
---|
| 16 | // You should have received a copy of the GNU General Public |
---|
| 17 | // License along with this work; if not, write to the Free Software |
---|
| 18 | // Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. |
---|
| 19 | // |
---|
| 20 | // ========== Copyright Header End ============================================ |
---|
| 21 | //////////////////////////////////////////////////////////////////////// |
---|
| 22 | /* |
---|
| 23 | // Module Name: sparc_tlu_intdp |
---|
| 24 | // Description: |
---|
| 25 | // Contains the code for receiving interrupts from the crossbar, |
---|
| 26 | // and sending interrupts out to other processors through the corssbar. |
---|
| 27 | // The interrupt receive register (INRR, asi=0x49/VA=0), incoming |
---|
| 28 | // vector register (INVR, asi=0x7f/VA=0x40), and interrupt vector |
---|
| 29 | // dispatch register (INDR, asi=0x77/VA=0) are implemented in this |
---|
| 30 | // block. This block also initiates thread reset/wake up when a |
---|
| 31 | // reset packet is received. |
---|
| 32 | // |
---|
| 33 | */ |
---|
| 34 | |
---|
| 35 | //////////////////////////////////////////////////////////////////////// |
---|
| 36 | // Global header file includes |
---|
| 37 | //////////////////////////////////////////////////////////////////////// |
---|
| 38 | `include "iop.h" |
---|
| 39 | |
---|
| 40 | //////////////////////////////////////////////////////////////////////// |
---|
| 41 | // Local header file includes / local defines |
---|
| 42 | //////////////////////////////////////////////////////////////////////// |
---|
| 43 | `include "tlu.h" |
---|
| 44 | // |
---|
| 45 | // modved defines to tlu.h |
---|
| 46 | /* |
---|
| 47 | `define INT_VEC_HI 5 |
---|
| 48 | `define INT_VEC_LO 0 |
---|
| 49 | `define INT_THR_HI 12 |
---|
| 50 | `define INT_THR_LO 8 |
---|
| 51 | `define INT_TYPE_HI 17 |
---|
| 52 | `define INT_TYPE_LO 16 |
---|
| 53 | */ |
---|
| 54 | |
---|
| 55 | module sparc_tlu_intdp (/*AUTOARG*/ |
---|
| 56 | // Outputs |
---|
| 57 | int_pending_i2_l, ind_inc_thrid_i1, // indr_inc_rst_pkt, |
---|
| 58 | ind_inc_type_i1, tlu_lsu_int_ldxa_data_w2, int_tlu_rstid_m, |
---|
| 59 | tlu_lsu_pcxpkt, so, |
---|
| 60 | // Inputs |
---|
| 61 | // |
---|
| 62 | // modified to abide to the Niagara reset methodology |
---|
| 63 | // clk, se, si, reset, lsu_tlu_intpkt, lsu_tlu_st_rs3_data_g, |
---|
| 64 | rclk, se, si, tlu_rst_l, lsu_tlu_st_rs3_data_g, // lsu_tlu_intpkt, |
---|
| 65 | inc_ind_ld_int_i1, inc_ind_rstthr_i1, inc_ind_asi_thr, |
---|
| 66 | inc_ind_asi_wr_indr, inc_ind_indr_grant, // inc_ind_asi_inrr, |
---|
| 67 | inc_ind_thr_m, inc_ind_asi_wr_inrr, inc_ind_asi_rd_invr, |
---|
| 68 | inc_indr_req_valid, inc_indr_req_thrid, tlu_asi_rdata_mxsel_g, |
---|
| 69 | tlu_asi_queue_rdata_g, tlu_scpd_asi_rdata_g, lsu_ind_intpkt_id, |
---|
| 70 | lsu_ind_intpkt_type, lsu_ind_intpkt_thr |
---|
| 71 | ); |
---|
| 72 | |
---|
| 73 | // |
---|
| 74 | // modified to abide to the Niagara reset methodology |
---|
| 75 | // input clk, se, si, reset; |
---|
| 76 | input rclk, se, si, tlu_rst_l; |
---|
| 77 | |
---|
| 78 | // from lsu |
---|
| 79 | // input [17:0] lsu_tlu_intpkt; // int pkt from cpx |
---|
| 80 | input [63:0] lsu_tlu_st_rs3_data_g; // write data for int regs |
---|
| 81 | |
---|
| 82 | // select lines from int_ctl |
---|
| 83 | input [3:0] inc_ind_ld_int_i1; // ld ext interrupt to inrr |
---|
| 84 | input [3:0] inc_ind_rstthr_i1; |
---|
| 85 | |
---|
| 86 | // changing the select from inverting to non-inverting for grape |
---|
| 87 | // input [3:0] inc_ind_asi_thr_l; // thread issuing asi command |
---|
| 88 | input [3:0] inc_ind_asi_thr; // thread issuing asi command |
---|
| 89 | input [3:0] inc_ind_asi_wr_indr; // write INDR |
---|
| 90 | // convert the signal to non-inverting version for grape |
---|
| 91 | // input [3:0] inc_ind_indr_grant_l; // transmit INDR to PCX |
---|
| 92 | input [3:0] inc_ind_indr_grant; // transmit INDR to PCX |
---|
| 93 | // obsolete input |
---|
| 94 | // input inc_ind_asi_inrr; // read INRR |
---|
| 95 | // convert the signal to non-inverting version for grape |
---|
| 96 | // input [3:0] inc_ind_thr_m_l; |
---|
| 97 | input [3:0] inc_ind_thr_m; |
---|
| 98 | |
---|
| 99 | // other controls |
---|
| 100 | input [3:0] inc_ind_asi_wr_inrr; // write INRR |
---|
| 101 | input [3:0] inc_ind_asi_rd_invr; // read INVR (reset corr bit in INRR) |
---|
| 102 | |
---|
| 103 | // indr request |
---|
| 104 | input inc_indr_req_valid; // valid value in INDR, i.e make req |
---|
| 105 | input [1:0] inc_indr_req_thrid; // thread making request |
---|
| 106 | // |
---|
| 107 | // asi rdata mux select |
---|
| 108 | input [3:0] tlu_asi_rdata_mxsel_g; |
---|
| 109 | // asi data from other blocks |
---|
| 110 | input [`TLU_SCPD_DATA_WIDTH-1:0] tlu_scpd_asi_rdata_g; |
---|
| 111 | input [`TLU_ASI_QUE_WIDTH-1:0] tlu_asi_queue_rdata_g; |
---|
| 112 | input [4:0] lsu_ind_intpkt_thr; |
---|
| 113 | input [1:0] lsu_ind_intpkt_type; |
---|
| 114 | input [5:0] lsu_ind_intpkt_id; |
---|
| 115 | |
---|
| 116 | // to int ctl |
---|
| 117 | output [3:0] int_pending_i2_l; // interrupt still pending |
---|
| 118 | // output indr_inc_rst_pkt; |
---|
| 119 | |
---|
| 120 | output [4:0] ind_inc_thrid_i1; |
---|
| 121 | output [1:0] ind_inc_type_i1; |
---|
| 122 | |
---|
| 123 | // to outside world |
---|
| 124 | output [63:0] tlu_lsu_int_ldxa_data_w2; // read data from asi regs |
---|
| 125 | output [5:0] int_tlu_rstid_m; |
---|
| 126 | |
---|
| 127 | output [25:0] tlu_lsu_pcxpkt; // pcxpkt for inter processor int |
---|
| 128 | |
---|
| 129 | output so; |
---|
| 130 | |
---|
| 131 | // local signals |
---|
| 132 | // |
---|
| 133 | // added to abide to the Niagara reset methodology |
---|
| 134 | wire local_rst; // local reset signal |
---|
| 135 | // |
---|
| 136 | wire [63:0] int_tlu_asi_data; // read data from int regs |
---|
| 137 | // interrupt and reset id |
---|
| 138 | wire [5:0] int_id_i1; |
---|
| 139 | wire [5:0] t0_rstid_i2, |
---|
| 140 | t1_rstid_i2, |
---|
| 141 | t2_rstid_i2, |
---|
| 142 | t3_rstid_i2, |
---|
| 143 | next_t0_rstid_i1, |
---|
| 144 | next_t1_rstid_i1, |
---|
| 145 | next_t2_rstid_i1, |
---|
| 146 | next_t3_rstid_i1; |
---|
| 147 | |
---|
| 148 | // Interrupt receive register |
---|
| 149 | wire [63:0] inrr_dec_i1, |
---|
| 150 | inrr_rd_data_i2; |
---|
| 151 | |
---|
| 152 | wire [63:0] t0_inrr_i2, |
---|
| 153 | t1_inrr_i2, |
---|
| 154 | t2_inrr_i2, |
---|
| 155 | t3_inrr_i2, |
---|
| 156 | t0_inrr_aw_i2, |
---|
| 157 | t1_inrr_aw_i2, |
---|
| 158 | t2_inrr_aw_i2, |
---|
| 159 | t3_inrr_aw_i2, |
---|
| 160 | t0_inrr_arw_i1, |
---|
| 161 | t1_inrr_arw_i1, |
---|
| 162 | t2_inrr_arw_i1, |
---|
| 163 | t3_inrr_arw_i1, |
---|
| 164 | next_t0_inrr_i1, |
---|
| 165 | next_t1_inrr_i1, |
---|
| 166 | next_t2_inrr_i1, |
---|
| 167 | next_t3_inrr_i1; |
---|
| 168 | |
---|
| 169 | wire [63:0] new_t0_inrr_i1, |
---|
| 170 | new_t1_inrr_i1, |
---|
| 171 | new_t2_inrr_i1, |
---|
| 172 | new_t3_inrr_i1; |
---|
| 173 | |
---|
| 174 | // clear interrupt through asi |
---|
| 175 | wire [63:0] t0_asi_wr_data, |
---|
| 176 | t1_asi_wr_data, |
---|
| 177 | t2_asi_wr_data, |
---|
| 178 | t3_asi_wr_data; |
---|
| 179 | |
---|
| 180 | // interrupt vector |
---|
| 181 | wire [5:0] t0_invr_i3, |
---|
| 182 | t1_invr_i3, |
---|
| 183 | t2_invr_i3, |
---|
| 184 | t3_invr_i3, |
---|
| 185 | t0_invr_i2, |
---|
| 186 | t1_invr_i2, |
---|
| 187 | t2_invr_i2, |
---|
| 188 | t3_invr_i2; |
---|
| 189 | wire [5:0] invr_rd_data_i3; |
---|
| 190 | |
---|
| 191 | // highest priority interrupt |
---|
| 192 | wire [63:0] pe_ivec_i3, |
---|
| 193 | t0_pe_ivec_i3, |
---|
| 194 | t1_pe_ivec_i3, |
---|
| 195 | t2_pe_ivec_i3, |
---|
| 196 | t3_pe_ivec_i3; |
---|
| 197 | |
---|
| 198 | // interrupt dispatch |
---|
| 199 | // removed the obsolete bits |
---|
| 200 | // wire [12:0] indr_wr_pkt; |
---|
| 201 | wire [10:0] indr_wr_pkt; |
---|
| 202 | |
---|
| 203 | // removed the obsolete bits |
---|
| 204 | // wire [12:0] indr_pcxpkt, |
---|
| 205 | wire [10:0] indr_pcxpkt, |
---|
| 206 | t0_indr, |
---|
| 207 | t1_indr, |
---|
| 208 | t2_indr, |
---|
| 209 | t3_indr, |
---|
| 210 | t0_indr_next, |
---|
| 211 | t1_indr_next, |
---|
| 212 | t2_indr_next, |
---|
| 213 | t3_indr_next; |
---|
| 214 | // |
---|
| 215 | // local clock |
---|
| 216 | wire clk; |
---|
| 217 | |
---|
| 218 | // |
---|
| 219 | // Code Starts Here |
---|
| 220 | // |
---|
| 221 | //---------------------------------------------------------------------- |
---|
| 222 | // creating local clock |
---|
| 223 | //---------------------------------------------------------------------- |
---|
| 224 | assign clk = rclk; |
---|
| 225 | |
---|
| 226 | //---------------------------------------------------------------------- |
---|
| 227 | // Interrupt Receive |
---|
| 228 | //---------------------------------------------------------------------- |
---|
| 229 | // |
---|
| 230 | // create local reset signal |
---|
| 231 | assign local_rst = ~tlu_rst_l; |
---|
| 232 | |
---|
| 233 | // I1 Stage |
---|
| 234 | // decode interrupt vector |
---|
| 235 | // modified due to interface clean-up |
---|
| 236 | /* |
---|
| 237 | assign int_id_i1 = lsu_tlu_intpkt[`INT_VEC_HI:`INT_VEC_LO]; |
---|
| 238 | assign ind_inc_type_i1 = lsu_tlu_intpkt[`INT_TYPE_HI:`INT_TYPE_LO]; |
---|
| 239 | assign ind_inc_thrid_i1 = lsu_tlu_intpkt[`INT_THR_HI:`INT_THR_LO]; |
---|
| 240 | */ |
---|
| 241 | assign int_id_i1[5:0] = lsu_ind_intpkt_id[5:0]; |
---|
| 242 | assign ind_inc_type_i1[1:0] = lsu_ind_intpkt_type[1:0]; |
---|
| 243 | assign ind_inc_thrid_i1[4:0] = lsu_ind_intpkt_thr[4:0]; |
---|
| 244 | |
---|
| 245 | // rstid enable mux |
---|
| 246 | dp_mux2es #6 rid_mux0(.dout (next_t0_rstid_i1[5:0]), |
---|
| 247 | .in0 (t0_rstid_i2[5:0]), |
---|
| 248 | .in1 (int_id_i1[5:0]), |
---|
| 249 | .sel (inc_ind_rstthr_i1[0])); |
---|
| 250 | |
---|
| 251 | `ifdef FPGA_SYN_1THREAD |
---|
| 252 | dff_s #6 rid0_reg(.din (next_t0_rstid_i1[5:0]), |
---|
| 253 | .q (t0_rstid_i2[5:0]), |
---|
| 254 | .clk (clk), |
---|
| 255 | .se (se), .si(), .so()); |
---|
| 256 | assign int_tlu_rstid_m[5:0] = t0_rstid_i2[5:0]; |
---|
| 257 | |
---|
| 258 | `else |
---|
| 259 | |
---|
| 260 | dp_mux2es #6 rid_mux1(.dout (next_t1_rstid_i1[5:0]), |
---|
| 261 | .in0 (t1_rstid_i2[5:0]), |
---|
| 262 | .in1 (int_id_i1[5:0]), |
---|
| 263 | .sel (inc_ind_rstthr_i1[1])); |
---|
| 264 | |
---|
| 265 | dp_mux2es #6 rid_mux2(.dout (next_t2_rstid_i1[5:0]), |
---|
| 266 | .in0 (t2_rstid_i2[5:0]), |
---|
| 267 | .in1 (int_id_i1[5:0]), |
---|
| 268 | .sel (inc_ind_rstthr_i1[2])); |
---|
| 269 | |
---|
| 270 | dp_mux2es #6 rid_mux3(.dout (next_t3_rstid_i1[5:0]), |
---|
| 271 | .in0 (t3_rstid_i2[5:0]), |
---|
| 272 | .in1 (int_id_i1[5:0]), |
---|
| 273 | .sel (inc_ind_rstthr_i1[3])); |
---|
| 274 | |
---|
| 275 | // rst id flops |
---|
| 276 | dff_s #6 rid0_reg(.din (next_t0_rstid_i1[5:0]), |
---|
| 277 | .q (t0_rstid_i2[5:0]), |
---|
| 278 | .clk (clk), |
---|
| 279 | .se (se), .si(), .so()); |
---|
| 280 | dff_s #6 rid1_reg(.din (next_t1_rstid_i1[5:0]), |
---|
| 281 | .q (t1_rstid_i2[5:0]), |
---|
| 282 | .clk (clk), |
---|
| 283 | .se (se), .si(), .so()); |
---|
| 284 | dff_s #6 rid2_reg(.din (next_t2_rstid_i1[5:0]), |
---|
| 285 | .q (t2_rstid_i2[5:0]), |
---|
| 286 | .clk (clk), |
---|
| 287 | .se (se), .si(), .so()); |
---|
| 288 | dff_s #6 rid3_reg(.din (next_t3_rstid_i1[5:0]), |
---|
| 289 | .q (t3_rstid_i2[5:0]), |
---|
| 290 | .clk (clk), |
---|
| 291 | .se (se), .si(), .so()); |
---|
| 292 | |
---|
| 293 | // rstid to tlu in M stage |
---|
| 294 | // changing the select from inverting to non-inverting for grape |
---|
| 295 | /* |
---|
| 296 | dp_mux4ds #6 tlurid_mux(.dout (int_tlu_rstid_m[5:0]), |
---|
| 297 | .in0 (t0_rstid_i2[5:0]), |
---|
| 298 | .in1 (t1_rstid_i2[5:0]), |
---|
| 299 | .in2 (t2_rstid_i2[5:0]), |
---|
| 300 | .in3 (t3_rstid_i2[5:0]), |
---|
| 301 | .sel0_l (inc_ind_thr_m_l[0]), |
---|
| 302 | .sel1_l (inc_ind_thr_m_l[1]), |
---|
| 303 | .sel2_l (inc_ind_thr_m_l[2]), |
---|
| 304 | .sel3_l (inc_ind_thr_m_l[3])); |
---|
| 305 | */ |
---|
| 306 | dp_mux4ds #6 tlurid_mux(.dout (int_tlu_rstid_m[5:0]), |
---|
| 307 | .in0 (t0_rstid_i2[5:0]), |
---|
| 308 | .in1 (t1_rstid_i2[5:0]), |
---|
| 309 | .in2 (t2_rstid_i2[5:0]), |
---|
| 310 | .in3 (t3_rstid_i2[5:0]), |
---|
| 311 | .sel0_l (~inc_ind_thr_m[0]), |
---|
| 312 | .sel1_l (~inc_ind_thr_m[1]), |
---|
| 313 | .sel2_l (~inc_ind_thr_m[2]), |
---|
| 314 | .sel3_l (~inc_ind_thr_m[3])); |
---|
| 315 | |
---|
| 316 | `endif // !`ifdef FPGA_SYN_1THREAD |
---|
| 317 | |
---|
| 318 | sparc_tlu_dec64 iv_dec(.in (int_id_i1[5:0]), |
---|
| 319 | .out (inrr_dec_i1[63:0])); |
---|
| 320 | |
---|
| 321 | // merge decoded interrupt vector with inrr |
---|
| 322 | assign new_t0_inrr_i1 = inrr_dec_i1 | t0_inrr_arw_i1; |
---|
| 323 | assign new_t1_inrr_i1 = inrr_dec_i1 | t1_inrr_arw_i1; |
---|
| 324 | assign new_t2_inrr_i1 = inrr_dec_i1 | t2_inrr_arw_i1; |
---|
| 325 | assign new_t3_inrr_i1 = inrr_dec_i1 | t3_inrr_arw_i1; |
---|
| 326 | |
---|
| 327 | // enable mux to load new interrupt to INRR |
---|
| 328 | dp_mux2es #64 inrr_en_mux0(.dout (next_t0_inrr_i1[63:0]), |
---|
| 329 | .in0 (t0_inrr_arw_i1[63:0]), |
---|
| 330 | .in1 (new_t0_inrr_i1[63:0]), |
---|
| 331 | .sel (inc_ind_ld_int_i1[0])); |
---|
| 332 | `ifdef FPGA_SYN_1THREAD |
---|
| 333 | // interrupt receive register (INRR) |
---|
| 334 | // change to dff -- software will reset before IE turns on |
---|
| 335 | dffr_s #64 t0_inrr (.din (next_t0_inrr_i1[63:0]), |
---|
| 336 | .q (t0_inrr_i2[63:0]), |
---|
| 337 | .clk (clk), |
---|
| 338 | // |
---|
| 339 | // modified to abide to the Niagara reset methodology |
---|
| 340 | // .rst (reset), |
---|
| 341 | .rst (local_rst), |
---|
| 342 | .se (se), .si(), .so()); |
---|
| 343 | assign inrr_rd_data_i2[63:0] = t0_inrr_i2[63:0]; |
---|
| 344 | |
---|
| 345 | `else |
---|
| 346 | |
---|
| 347 | dp_mux2es #64 inrr_en_mux1(.dout (next_t1_inrr_i1[63:0]), |
---|
| 348 | .in0 (t1_inrr_arw_i1[63:0]), |
---|
| 349 | .in1 (new_t1_inrr_i1[63:0]), |
---|
| 350 | .sel (inc_ind_ld_int_i1[1])); |
---|
| 351 | dp_mux2es #64 inrr_en_mux2(.dout (next_t2_inrr_i1[63:0]), |
---|
| 352 | .in0 (t2_inrr_arw_i1[63:0]), |
---|
| 353 | .in1 (new_t2_inrr_i1[63:0]), |
---|
| 354 | .sel (inc_ind_ld_int_i1[2])); |
---|
| 355 | dp_mux2es #64 inrr_en_mux3(.dout (next_t3_inrr_i1[63:0]), |
---|
| 356 | .in0 (t3_inrr_arw_i1[63:0]), |
---|
| 357 | .in1 (new_t3_inrr_i1[63:0]), |
---|
| 358 | .sel (inc_ind_ld_int_i1[3])); |
---|
| 359 | |
---|
| 360 | // interrupt receive register (INRR) |
---|
| 361 | // change to dff -- software will reset before IE turns on |
---|
| 362 | dffr_s #64 t0_inrr (.din (next_t0_inrr_i1[63:0]), |
---|
| 363 | .q (t0_inrr_i2[63:0]), |
---|
| 364 | .clk (clk), |
---|
| 365 | // |
---|
| 366 | // modified to abide to the Niagara reset methodology |
---|
| 367 | // .rst (reset), |
---|
| 368 | .rst (local_rst), |
---|
| 369 | .se (se), .si(), .so()); |
---|
| 370 | dffr_s #64 t1_inrr (.din (next_t1_inrr_i1[63:0]), |
---|
| 371 | .q (t1_inrr_i2[63:0]), |
---|
| 372 | .clk (clk), |
---|
| 373 | // |
---|
| 374 | // modified to abide to the Niagara reset methodology |
---|
| 375 | // .rst (reset), |
---|
| 376 | .rst (local_rst), |
---|
| 377 | .se (se), .si(), .so()); |
---|
| 378 | dffr_s #64 t2_inrr (.din (next_t2_inrr_i1[63:0]), |
---|
| 379 | .q (t2_inrr_i2[63:0]), |
---|
| 380 | .clk (clk), |
---|
| 381 | // |
---|
| 382 | // modified to abide to the Niagara reset methodology |
---|
| 383 | // .rst (reset), |
---|
| 384 | .rst (local_rst), |
---|
| 385 | .se (se), .si(), .so()); |
---|
| 386 | dffr_s #64 t3_inrr (.din (next_t3_inrr_i1[63:0]), |
---|
| 387 | .q (t3_inrr_i2[63:0]), |
---|
| 388 | .clk (clk), |
---|
| 389 | // |
---|
| 390 | // modified to abide to the Niagara reset methodology |
---|
| 391 | // .rst (reset), |
---|
| 392 | .rst (local_rst), |
---|
| 393 | .se (se), .si(), .so()); |
---|
| 394 | |
---|
| 395 | // I2 Stage |
---|
| 396 | // read out INRR to asi |
---|
| 397 | // changing the select from inverting to non-inverting for grape |
---|
| 398 | /* |
---|
| 399 | dp_mux4ds #64 inrr_rd_mux(.dout (inrr_rd_data_i2[63:0]), |
---|
| 400 | .in0 (t0_inrr_i2[63:0]), |
---|
| 401 | .in1 (t1_inrr_i2[63:0]), |
---|
| 402 | .in2 (t2_inrr_i2[63:0]), |
---|
| 403 | .in3 (t3_inrr_i2[63:0]), |
---|
| 404 | .sel0_l (inc_ind_asi_thr_l[0]), |
---|
| 405 | .sel1_l (inc_ind_asi_thr_l[1]), |
---|
| 406 | .sel2_l (inc_ind_asi_thr_l[2]), |
---|
| 407 | .sel3_l (inc_ind_asi_thr_l[3])); |
---|
| 408 | */ |
---|
| 409 | dp_mux4ds #64 inrr_rd_mux(.dout (inrr_rd_data_i2[63:0]), |
---|
| 410 | .in0 (t0_inrr_i2[63:0]), |
---|
| 411 | .in1 (t1_inrr_i2[63:0]), |
---|
| 412 | .in2 (t2_inrr_i2[63:0]), |
---|
| 413 | .in3 (t3_inrr_i2[63:0]), |
---|
| 414 | .sel0_l (~inc_ind_asi_thr[0]), |
---|
| 415 | .sel1_l (~inc_ind_asi_thr[1]), |
---|
| 416 | .sel2_l (~inc_ind_asi_thr[2]), |
---|
| 417 | .sel3_l (~inc_ind_asi_thr[3])); |
---|
| 418 | |
---|
| 419 | `endif // !`ifdef FPGA_SYN_1THREAD |
---|
| 420 | |
---|
| 421 | // signal interrupt pending |
---|
| 422 | sparc_tlu_zcmp64 zcmp0(.in (t0_inrr_i2[63:0]), |
---|
| 423 | .zero (int_pending_i2_l[0])); |
---|
| 424 | |
---|
| 425 | `ifdef FPGA_SYN_1THREAD |
---|
| 426 | assign t0_asi_wr_data = ~(~lsu_tlu_st_rs3_data_g & |
---|
| 427 | {64{inc_ind_asi_wr_inrr[0]}}); |
---|
| 428 | assign t0_inrr_aw_i2 = t0_inrr_i2 & t0_asi_wr_data; |
---|
| 429 | sparc_tlu_penc64 t0_invr_penc(.in (t0_inrr_i2[63:0]), |
---|
| 430 | .out (t0_invr_i2[5:0])); |
---|
| 431 | dff_s #6 t0_invr (.din (t0_invr_i2[5:0]), |
---|
| 432 | .q (t0_invr_i3[5:0]), |
---|
| 433 | .clk (clk), |
---|
| 434 | .se (se), .si(), .so()); |
---|
| 435 | assign invr_rd_data_i3[5:0] = t0_invr_i3[5:0]; |
---|
| 436 | |
---|
| 437 | `else |
---|
| 438 | |
---|
| 439 | sparc_tlu_zcmp64 zcmp1(.in (t1_inrr_i2[63:0]), |
---|
| 440 | .zero (int_pending_i2_l[1])); |
---|
| 441 | sparc_tlu_zcmp64 zcmp2(.in (t2_inrr_i2[63:0]), |
---|
| 442 | .zero (int_pending_i2_l[2])); |
---|
| 443 | sparc_tlu_zcmp64 zcmp3(.in (t3_inrr_i2[63:0]), |
---|
| 444 | .zero (int_pending_i2_l[3])); |
---|
| 445 | |
---|
| 446 | // write data -- only zeros may be written to the INRR. An attempt |
---|
| 447 | // to write 1 is ignored. |
---|
| 448 | // Force to all 1 if no write |
---|
| 449 | assign t0_asi_wr_data = ~(~lsu_tlu_st_rs3_data_g & |
---|
| 450 | {64{inc_ind_asi_wr_inrr[0]}}); |
---|
| 451 | assign t1_asi_wr_data = ~(~lsu_tlu_st_rs3_data_g & |
---|
| 452 | {64{inc_ind_asi_wr_inrr[1]}}); |
---|
| 453 | assign t2_asi_wr_data = ~(~lsu_tlu_st_rs3_data_g & |
---|
| 454 | {64{inc_ind_asi_wr_inrr[2]}}); |
---|
| 455 | assign t3_asi_wr_data = ~(~lsu_tlu_st_rs3_data_g & |
---|
| 456 | {64{inc_ind_asi_wr_inrr[3]}}); |
---|
| 457 | |
---|
| 458 | assign t0_inrr_aw_i2 = t0_inrr_i2 & t0_asi_wr_data; |
---|
| 459 | assign t1_inrr_aw_i2 = t1_inrr_i2 & t1_asi_wr_data; |
---|
| 460 | assign t2_inrr_aw_i2 = t2_inrr_i2 & t2_asi_wr_data; |
---|
| 461 | assign t3_inrr_aw_i2 = t3_inrr_i2 & t3_asi_wr_data; |
---|
| 462 | |
---|
| 463 | // priority encode INRR to 6 bits to get INVR |
---|
| 464 | // b63 has the highest priority |
---|
| 465 | sparc_tlu_penc64 t0_invr_penc(.in (t0_inrr_i2[63:0]), |
---|
| 466 | .out (t0_invr_i2[5:0])); |
---|
| 467 | sparc_tlu_penc64 t1_invr_penc(.in (t1_inrr_i2[63:0]), |
---|
| 468 | .out (t1_invr_i2[5:0])); |
---|
| 469 | sparc_tlu_penc64 t2_invr_penc(.in (t2_inrr_i2[63:0]), |
---|
| 470 | .out (t2_invr_i2[5:0])); |
---|
| 471 | sparc_tlu_penc64 t3_invr_penc(.in (t3_inrr_i2[63:0]), |
---|
| 472 | .out (t3_invr_i2[5:0])); |
---|
| 473 | |
---|
| 474 | // Interrupt Vector Register (INVR) |
---|
| 475 | // Cannot write to INVR |
---|
| 476 | dff_s #6 t0_invr (.din (t0_invr_i2[5:0]), |
---|
| 477 | .q (t0_invr_i3[5:0]), |
---|
| 478 | .clk (clk), |
---|
| 479 | .se (se), .si(), .so()); |
---|
| 480 | dff_s #6 t1_invr (.din (t1_invr_i2[5:0]), |
---|
| 481 | .q (t1_invr_i3[5:0]), |
---|
| 482 | .clk (clk), |
---|
| 483 | .se (se), .si(), .so()); |
---|
| 484 | dff_s #6 t2_invr (.din (t2_invr_i2[5:0]), |
---|
| 485 | .q (t2_invr_i3[5:0]), |
---|
| 486 | .clk (clk), |
---|
| 487 | .se (se), .si(), .so()); |
---|
| 488 | dff_s #6 t3_invr (.din (t3_invr_i2[5:0]), |
---|
| 489 | .q (t3_invr_i3[5:0]), |
---|
| 490 | .clk (clk), |
---|
| 491 | .se (se), .si(), .so()); |
---|
| 492 | |
---|
| 493 | // I3 stage |
---|
| 494 | // read out to asi data |
---|
| 495 | // changing the select from inverting to non-inverting for grape |
---|
| 496 | /* |
---|
| 497 | dp_mux4ds #6 invr_rd_mux(.dout (invr_rd_data_i3[5:0]), |
---|
| 498 | .in0 (t0_invr_i3[5:0]), |
---|
| 499 | .in1 (t1_invr_i3[5:0]), |
---|
| 500 | .in2 (t2_invr_i3[5:0]), |
---|
| 501 | .in3 (t3_invr_i3[5:0]), |
---|
| 502 | .sel0_l (inc_ind_asi_thr_l[0]), |
---|
| 503 | .sel1_l (inc_ind_asi_thr_l[1]), |
---|
| 504 | .sel2_l (inc_ind_asi_thr_l[2]), |
---|
| 505 | .sel3_l (inc_ind_asi_thr_l[3])); |
---|
| 506 | */ |
---|
| 507 | dp_mux4ds #6 invr_rd_mux(.dout (invr_rd_data_i3[5:0]), |
---|
| 508 | .in0 (t0_invr_i3[5:0]), |
---|
| 509 | .in1 (t1_invr_i3[5:0]), |
---|
| 510 | .in2 (t2_invr_i3[5:0]), |
---|
| 511 | .in3 (t3_invr_i3[5:0]), |
---|
| 512 | .sel0_l (~inc_ind_asi_thr[0]), |
---|
| 513 | .sel1_l (~inc_ind_asi_thr[1]), |
---|
| 514 | .sel2_l (~inc_ind_asi_thr[2]), |
---|
| 515 | .sel3_l (~inc_ind_asi_thr[3])); |
---|
| 516 | `endif // !`ifdef FPGA_SYN_1THREAD |
---|
| 517 | |
---|
| 518 | // |
---|
| 519 | // modified for bug 2109 |
---|
| 520 | // asi rd data mux |
---|
| 521 | dp_mux4ds #(64) asi_rd_mux( |
---|
| 522 | .in0 ({58'b0, invr_rd_data_i3[5:0]}), |
---|
| 523 | .in1 (inrr_rd_data_i2[63:0]), |
---|
| 524 | .in2 (tlu_scpd_asi_rdata_g[`TLU_SCPD_DATA_WIDTH-1:0]), |
---|
| 525 | .in3 ({50'b0, tlu_asi_queue_rdata_g[`TLU_ASI_QUE_WIDTH-1:0],6'b0}), |
---|
| 526 | .sel0_l (~tlu_asi_rdata_mxsel_g[0]), |
---|
| 527 | .sel1_l (~tlu_asi_rdata_mxsel_g[1]), |
---|
| 528 | .sel2_l (~tlu_asi_rdata_mxsel_g[2]), |
---|
| 529 | .sel3_l (~tlu_asi_rdata_mxsel_g[3]), |
---|
| 530 | .dout (int_tlu_asi_data[63:0])); |
---|
| 531 | |
---|
| 532 | dff_s #(64) dff_tlu_lsu_int_ldxa_data_w2 ( |
---|
| 533 | .din (int_tlu_asi_data[63:0]), |
---|
| 534 | .q (tlu_lsu_int_ldxa_data_w2[63:0]), |
---|
| 535 | .clk (clk), |
---|
| 536 | .se (se), |
---|
| 537 | .si(), |
---|
| 538 | .so()); |
---|
| 539 | |
---|
| 540 | sparc_tlu_dec64 inrr_pe_dec(.in (invr_rd_data_i3[5:0]), |
---|
| 541 | .out (pe_ivec_i3[63:0])); |
---|
| 542 | |
---|
| 543 | // when INVR is read, zero out the corresponding bit in INRR |
---|
| 544 | assign t0_pe_ivec_i3 = pe_ivec_i3 & {64{inc_ind_asi_rd_invr[0]}}; |
---|
| 545 | assign t1_pe_ivec_i3 = pe_ivec_i3 & {64{inc_ind_asi_rd_invr[1]}}; |
---|
| 546 | assign t2_pe_ivec_i3 = pe_ivec_i3 & {64{inc_ind_asi_rd_invr[2]}}; |
---|
| 547 | assign t3_pe_ivec_i3 = pe_ivec_i3 & {64{inc_ind_asi_rd_invr[3]}}; |
---|
| 548 | |
---|
| 549 | assign t0_inrr_arw_i1 = t0_inrr_aw_i2 & ~t0_pe_ivec_i3; |
---|
| 550 | assign t1_inrr_arw_i1 = t1_inrr_aw_i2 & ~t1_pe_ivec_i3; |
---|
| 551 | assign t2_inrr_arw_i1 = t2_inrr_aw_i2 & ~t2_pe_ivec_i3; |
---|
| 552 | assign t3_inrr_arw_i1 = t3_inrr_aw_i2 & ~t3_pe_ivec_i3; |
---|
| 553 | |
---|
| 554 | //---------------------------------------------------------------------- |
---|
| 555 | // Interrupt Dispatch |
---|
| 556 | //---------------------------------------------------------------------- |
---|
| 557 | // modified to remove the unused bits |
---|
| 558 | // |
---|
| 559 | // assign indr_wr_pkt = {lsu_tlu_st_rs3_data_g[`INT_TYPE_HI:`INT_TYPE_LO], |
---|
| 560 | assign indr_wr_pkt = {lsu_tlu_st_rs3_data_g[`INT_THR_HI:`INT_THR_LO], |
---|
| 561 | lsu_tlu_st_rs3_data_g[`INT_VEC_HI:`INT_VEC_LO]}; |
---|
| 562 | // |
---|
| 563 | // removed for timing |
---|
| 564 | // assign indr_inc_rst_pkt = lsu_tlu_st_rs3_data_g[`INT_TYPE_HI] | |
---|
| 565 | // lsu_tlu_st_rs3_data_g[`INT_TYPE_LO]; |
---|
| 566 | |
---|
| 567 | dp_mux2es #11 t0_indr_mux(.dout (t0_indr_next[10:0]), |
---|
| 568 | .in0 (t0_indr[10:0]), |
---|
| 569 | .in1 (indr_wr_pkt[10:0]), |
---|
| 570 | .sel (inc_ind_asi_wr_indr[0])); |
---|
| 571 | `ifdef FPGA_SYN_1THREAD |
---|
| 572 | dff_s #11 t0_indr_reg(.din (t0_indr_next[10:0]), |
---|
| 573 | .q (t0_indr[10:0]), |
---|
| 574 | .clk (clk), |
---|
| 575 | .se (se), .si(), .so()); |
---|
| 576 | assign indr_pcxpkt[10:0] = t0_indr[10:0]; |
---|
| 577 | |
---|
| 578 | `else |
---|
| 579 | |
---|
| 580 | dp_mux2es #11 t1_indr_mux(.dout (t1_indr_next[10:0]), |
---|
| 581 | .in0 (t1_indr[10:0]), |
---|
| 582 | .in1 (indr_wr_pkt[10:0]), |
---|
| 583 | .sel (inc_ind_asi_wr_indr[1])); |
---|
| 584 | dp_mux2es #11 t2_indr_mux(.dout (t2_indr_next[10:0]), |
---|
| 585 | .in0 (t2_indr[10:0]), |
---|
| 586 | .in1 (indr_wr_pkt[10:0]), |
---|
| 587 | .sel (inc_ind_asi_wr_indr[2])); |
---|
| 588 | dp_mux2es #11 t3_indr_mux(.dout (t3_indr_next[10:0]), |
---|
| 589 | .in0 (t3_indr[10:0]), |
---|
| 590 | .in1 (indr_wr_pkt[10:0]), |
---|
| 591 | .sel (inc_ind_asi_wr_indr[3])); |
---|
| 592 | |
---|
| 593 | dff_s #11 t0_indr_reg(.din (t0_indr_next[10:0]), |
---|
| 594 | .q (t0_indr[10:0]), |
---|
| 595 | .clk (clk), |
---|
| 596 | .se (se), .si(), .so()); |
---|
| 597 | dff_s #11 t1_indr_reg(.din (t1_indr_next[10:0]), |
---|
| 598 | .q (t1_indr[10:0]), |
---|
| 599 | .clk (clk), |
---|
| 600 | .se (se), .si(), .so()); |
---|
| 601 | dff_s #11 t2_indr_reg(.din (t2_indr_next[10:0]), |
---|
| 602 | .q (t2_indr[10:0]), |
---|
| 603 | .clk (clk), |
---|
| 604 | .se (se), .si(), .so()); |
---|
| 605 | dff_s #11 t3_indr_reg(.din (t3_indr_next[10:0]), |
---|
| 606 | .q (t3_indr[10:0]), |
---|
| 607 | .clk (clk), |
---|
| 608 | .se (se), .si(), .so()); |
---|
| 609 | |
---|
| 610 | // changing the select from inverting to non-inverting for grape |
---|
| 611 | /* |
---|
| 612 | dp_mux4ds #13 int_dsp_mux(.dout (indr_pcxpkt[12:0]), |
---|
| 613 | .in0 (t0_indr[12:0]), |
---|
| 614 | .in1 (t1_indr[12:0]), |
---|
| 615 | .in2 (t2_indr[12:0]), |
---|
| 616 | .in3 (t3_indr[12:0]), |
---|
| 617 | .sel0_l (inc_ind_indr_grant_l[0]), |
---|
| 618 | .sel1_l (inc_ind_indr_grant_l[1]), |
---|
| 619 | .sel2_l (inc_ind_indr_grant_l[2]), |
---|
| 620 | .sel3_l (inc_ind_indr_grant_l[3])); |
---|
| 621 | */ |
---|
| 622 | dp_mux4ds #11 int_dsp_mux(.dout (indr_pcxpkt[10:0]), |
---|
| 623 | .in0 (t0_indr[10:0]), |
---|
| 624 | .in1 (t1_indr[10:0]), |
---|
| 625 | .in2 (t2_indr[10:0]), |
---|
| 626 | .in3 (t3_indr[10:0]), |
---|
| 627 | .sel0_l (~inc_ind_indr_grant[0]), |
---|
| 628 | .sel1_l (~inc_ind_indr_grant[1]), |
---|
| 629 | .sel2_l (~inc_ind_indr_grant[2]), |
---|
| 630 | .sel3_l (~inc_ind_indr_grant[3])); |
---|
| 631 | `endif // !`ifdef FPGA_SYN_1THREAD |
---|
| 632 | |
---|
| 633 | |
---|
| 634 | assign tlu_lsu_pcxpkt[25:0] = {inc_indr_req_valid, // 25 |
---|
| 635 | {`INT_RQ}, // 24:20 |
---|
| 636 | inc_indr_req_thrid[1:0], // 19:18 |
---|
| 637 | // indr_pcxpkt[12:11], -- cannot send rst |
---|
| 638 | {2'b00}, // 17:16 |
---|
| 639 | 3'b0, // 15:13 rsvd |
---|
| 640 | indr_pcxpkt[10:6], // 12:8 |
---|
| 641 | 2'b0, // 7:6 rsvd |
---|
| 642 | indr_pcxpkt[5:0]}; // 5:0 |
---|
| 643 | |
---|
| 644 | // TBD: |
---|
| 645 | // 1. disable sending of reset/nuke/resum packets from indr -- DONE 1/6 |
---|
| 646 | |
---|
| 647 | endmodule |
---|
| 648 | |
---|
| 649 | |
---|