xref: /XiangShan/src/main/scala/xiangshan/backend/fu/CSR.scala (revision d479a3a838f93713e8d569af098b6da7fc3c5905)
1package xiangshan.backend.fu
2
3import chisel3._
4import chisel3.ExcitingUtils.{ConnectionType, Debug}
5import chisel3.util._
6import utils._
7import xiangshan._
8import xiangshan.backend._
9import xiangshan.frontend.BPUCtrl
10import xiangshan.backend.fu.util._
11
12trait HasExceptionNO {
13  def instrAddrMisaligned = 0
14  def instrAccessFault    = 1
15  def illegalInstr        = 2
16  def breakPoint          = 3
17  def loadAddrMisaligned  = 4
18  def loadAccessFault     = 5
19  def storeAddrMisaligned = 6
20  def storeAccessFault    = 7
21  def ecallU              = 8
22  def ecallS              = 9
23  def ecallM              = 11
24  def instrPageFault      = 12
25  def loadPageFault       = 13
26  def storePageFault      = 15
27
28  val ExcPriority = Seq(
29    breakPoint, // TODO: different BP has different priority
30    instrPageFault,
31    instrAccessFault,
32    illegalInstr,
33    instrAddrMisaligned,
34    ecallM, ecallS, ecallU,
35    storePageFault,
36    loadPageFault,
37    storeAccessFault,
38    loadAccessFault,
39    storeAddrMisaligned,
40    loadAddrMisaligned
41  )
42  val frontendSet = List(
43    // instrAddrMisaligned,
44    instrAccessFault,
45    illegalInstr,
46    instrPageFault
47  )
48  val csrSet = List(
49    illegalInstr,
50    breakPoint,
51    ecallU,
52    ecallS,
53    ecallM
54  )
55  val loadUnitSet = List(
56    loadAddrMisaligned,
57    loadAccessFault,
58    loadPageFault
59  )
60  val storeUnitSet = List(
61    storeAddrMisaligned,
62    storeAccessFault,
63    storePageFault
64  )
65  val atomicsUnitSet = (loadUnitSet ++ storeUnitSet).distinct
66  val allPossibleSet = (frontendSet ++ csrSet ++ loadUnitSet ++ storeUnitSet).distinct
67  val csrWbCount = (0 until 16).map(i => if (csrSet.contains(i)) 1 else 0)
68  val loadWbCount = (0 until 16).map(i => if (loadUnitSet.contains(i)) 1 else 0)
69  val storeWbCount = (0 until 16).map(i => if (storeUnitSet.contains(i)) 1 else 0)
70  val atomicsWbCount = (0 until 16).map(i => if (atomicsUnitSet.contains(i)) 1 else 0)
71  val writebackCount = (0 until 16).map(i => csrWbCount(i) + atomicsWbCount(i) + loadWbCount(i) + 2 * storeWbCount(i))
72  def partialSelect(vec: Vec[Bool], select: Seq[Int], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] = {
73    if (dontCareBits) {
74      val new_vec = Wire(ExceptionVec())
75      new_vec := DontCare
76      select.map(i => new_vec(i) := vec(i))
77      return new_vec
78    }
79    else if (falseBits) {
80      val new_vec = Wire(ExceptionVec())
81      new_vec.map(_ := false.B)
82      select.map(i => new_vec(i) := vec(i))
83      return new_vec
84    }
85    else {
86      val new_vec = Wire(Vec(select.length, Bool()))
87      select.zipWithIndex.map{ case(s, i) => new_vec(i) := vec(s) }
88      return new_vec
89    }
90  }
91  def selectFrontend(vec: Vec[Bool], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] =
92    partialSelect(vec, frontendSet, dontCareBits, falseBits)
93  def selectCSR(vec: Vec[Bool], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] =
94    partialSelect(vec, csrSet, dontCareBits, falseBits)
95  def selectLoad(vec: Vec[Bool], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] =
96    partialSelect(vec, loadUnitSet, dontCareBits, falseBits)
97  def selectStore(vec: Vec[Bool], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] =
98    partialSelect(vec, storeUnitSet, dontCareBits, falseBits)
99  def selectAtomics(vec: Vec[Bool], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] =
100    partialSelect(vec, atomicsUnitSet, dontCareBits, falseBits)
101  def selectAll(vec: Vec[Bool], dontCareBits: Boolean = true, falseBits: Boolean = false): Vec[Bool] =
102    partialSelect(vec, allPossibleSet, dontCareBits, falseBits)
103}
104
105class FpuCsrIO extends XSBundle {
106  val fflags = Output(Valid(UInt(5.W)))
107  val isIllegal = Output(Bool())
108  val dirty_fs = Output(Bool())
109  val frm = Input(UInt(3.W))
110}
111
112
113class PerfCounterIO extends XSBundle {
114  val retiredInstr = Input(UInt(3.W))
115  val value = Input(UInt(XLEN.W))
116}
117
118class CSRFileIO extends XSBundle {
119  val hartId = Input(UInt(64.W))
120  // output (for func === CSROpType.jmp)
121  val perf = new PerfCounterIO
122  val isPerfCnt = Output(Bool())
123  // to FPU
124  val fpu = Flipped(new FpuCsrIO)
125  // from rob
126  val exception = Flipped(ValidIO(new ExceptionInfo))
127  // to ROB
128  val isXRet = Output(Bool())
129  val trapTarget = Output(UInt(VAddrBits.W))
130  val interrupt = Output(Bool())
131  // from LSQ
132  val memExceptionVAddr = Input(UInt(VAddrBits.W))
133  // from outside cpu,externalInterrupt
134  val externalInterrupt = new ExternalInterruptIO
135  // TLB
136  val tlb = Output(new TlbCsrBundle)
137  // Custom microarchiture ctrl signal
138  val customCtrl = Output(new CustomCSRCtrlIO)
139}
140
141class CSR extends FunctionUnit with HasCSRConst
142{
143  val csrio = IO(new CSRFileIO)
144  val difftestIO = IO(new Bundle() {
145    val intrNO = Output(UInt(64.W))
146    val cause = Output(UInt(64.W))
147    val priviledgeMode = Output(UInt(2.W))
148    val mstatus = Output(UInt(64.W))
149    val sstatus = Output(UInt(64.W))
150    val mepc = Output(UInt(64.W))
151    val sepc = Output(UInt(64.W))
152    val mtval = Output(UInt(64.W))
153    val stval = Output(UInt(64.W))
154    val mtvec = Output(UInt(64.W))
155    val stvec = Output(UInt(64.W))
156    val mcause = Output(UInt(64.W))
157    val scause = Output(UInt(64.W))
158    val satp = Output(UInt(64.W))
159    val mip = Output(UInt(64.W))
160    val mie = Output(UInt(64.W))
161    val mscratch = Output(UInt(64.W))
162    val sscratch = Output(UInt(64.W))
163    val mideleg = Output(UInt(64.W))
164    val medeleg = Output(UInt(64.W))
165  })
166  difftestIO <> DontCare
167
168  val cfIn = io.in.bits.uop.cf
169  val cfOut = Wire(new CtrlFlow)
170  cfOut := cfIn
171  val flushPipe = Wire(Bool())
172
173  val (valid, src1, src2, func) = (
174    io.in.valid,
175    io.in.bits.src(0),
176    io.in.bits.uop.ctrl.imm,
177    io.in.bits.uop.ctrl.fuOpType
178  )
179
180  // CSR define
181
182  class Priv extends Bundle {
183    val m = Output(Bool())
184    val h = Output(Bool())
185    val s = Output(Bool())
186    val u = Output(Bool())
187  }
188
189  val csrNotImplemented = RegInit(UInt(XLEN.W), 0.U)
190
191  class MstatusStruct extends Bundle {
192    val sd = Output(UInt(1.W))
193
194    val pad1 = if (XLEN == 64) Output(UInt(27.W)) else null
195    val sxl  = if (XLEN == 64) Output(UInt(2.W))  else null
196    val uxl  = if (XLEN == 64) Output(UInt(2.W))  else null
197    val pad0 = if (XLEN == 64) Output(UInt(9.W))  else Output(UInt(8.W))
198
199    val tsr = Output(UInt(1.W))
200    val tw = Output(UInt(1.W))
201    val tvm = Output(UInt(1.W))
202    val mxr = Output(UInt(1.W))
203    val sum = Output(UInt(1.W))
204    val mprv = Output(UInt(1.W))
205    val xs = Output(UInt(2.W))
206    val fs = Output(UInt(2.W))
207    val mpp = Output(UInt(2.W))
208    val hpp = Output(UInt(2.W))
209    val spp = Output(UInt(1.W))
210    val pie = new Priv
211    val ie = new Priv
212    assert(this.getWidth == XLEN)
213  }
214
215  class SatpStruct extends Bundle {
216    val mode = UInt(4.W)
217    val asid = UInt(16.W)
218    val ppn  = UInt(44.W)
219  }
220
221  class Interrupt extends Bundle {
222    val e = new Priv
223    val t = new Priv
224    val s = new Priv
225  }
226
227  // Machine-Level CSRs
228
229  val mtvec = RegInit(UInt(XLEN.W), 0.U)
230  val mcounteren = RegInit(UInt(XLEN.W), 0.U)
231  val mcause = RegInit(UInt(XLEN.W), 0.U)
232  val mtval = RegInit(UInt(XLEN.W), 0.U)
233  val mepc = Reg(UInt(XLEN.W))
234
235  val mie = RegInit(0.U(XLEN.W))
236  val mipWire = WireInit(0.U.asTypeOf(new Interrupt))
237  val mipReg  = RegInit(0.U.asTypeOf(new Interrupt).asUInt)
238  val mipFixMask = GenMask(9) | GenMask(5) | GenMask(1)
239  val mip = (mipWire.asUInt | mipReg).asTypeOf(new Interrupt)
240
241  def getMisaMxl(mxl: Int): UInt = {mxl.U << (XLEN-2)}.asUInt()
242  def getMisaExt(ext: Char): UInt = {1.U << (ext.toInt - 'a'.toInt)}.asUInt()
243  var extList = List('a', 's', 'i', 'u')
244  if (HasMExtension) { extList = extList :+ 'm' }
245  if (HasCExtension) { extList = extList :+ 'c' }
246  if (HasFPU) { extList = extList ++ List('f', 'd') }
247  val misaInitVal = getMisaMxl(2) | extList.foldLeft(0.U)((sum, i) => sum | getMisaExt(i)) //"h8000000000141105".U
248  val misa = RegInit(UInt(XLEN.W), misaInitVal)
249
250  // MXL = 2          | 0 | EXT = b 00 0000 0100 0001 0001 0000 0101
251  // (XLEN-1, XLEN-2) |   |(25, 0)  ZY XWVU TSRQ PONM LKJI HGFE DCBA
252
253  val mvendorid = RegInit(UInt(XLEN.W), 0.U) // this is a non-commercial implementation
254  val marchid = RegInit(UInt(XLEN.W), 0.U) // return 0 to indicate the field is not implemented
255  val mimpid = RegInit(UInt(XLEN.W), 0.U) // provides a unique encoding of the version of the processor implementation
256  val mhartid = RegInit(UInt(XLEN.W), csrio.hartId) // the hardware thread running the code
257  val mstatus = RegInit(UInt(XLEN.W), 0.U)
258
259  // mstatus Value Table
260  // | sd   |
261  // | pad1 |
262  // | sxl  | hardlinked to 10, use 00 to pass xv6 test
263  // | uxl  | hardlinked to 00
264  // | pad0 |
265  // | tsr  |
266  // | tw   |
267  // | tvm  |
268  // | mxr  |
269  // | sum  |
270  // | mprv |
271  // | xs   | 00 |
272  // | fs   | 00 |
273  // | mpp  | 00 |
274  // | hpp  | 00 |
275  // | spp  | 0 |
276  // | pie  | 0000 | pie.h is used as UBE
277  // | ie   | 0000 | uie hardlinked to 0, as N ext is not implemented
278
279  val mstatusStruct = mstatus.asTypeOf(new MstatusStruct)
280  def mstatusUpdateSideEffect(mstatus: UInt): UInt = {
281    val mstatusOld = WireInit(mstatus.asTypeOf(new MstatusStruct))
282    val mstatusNew = Cat(mstatusOld.xs === "b11".U || mstatusOld.fs === "b11".U, mstatus(XLEN-2, 0))
283    mstatusNew
284  }
285
286  val mstatusMask = (~ZeroExt((
287    GenMask(XLEN-2, 38) | GenMask(31, 23) | GenMask(10, 9) | GenMask(2) |
288    GenMask(37) | // MBE
289    GenMask(36) | // SBE
290    GenMask(6)    // UBE
291  ), 64)).asUInt()
292
293  val medeleg = RegInit(UInt(XLEN.W), 0.U)
294  val mideleg = RegInit(UInt(XLEN.W), 0.U)
295  val mscratch = RegInit(UInt(XLEN.W), 0.U)
296
297  val pmpcfg0 = RegInit(UInt(XLEN.W), 0.U)
298  val pmpcfg1 = RegInit(UInt(XLEN.W), 0.U)
299  val pmpcfg2 = RegInit(UInt(XLEN.W), 0.U)
300  val pmpcfg3 = RegInit(UInt(XLEN.W), 0.U)
301  val pmpaddr0 = RegInit(UInt(XLEN.W), 0.U)
302  val pmpaddr1 = RegInit(UInt(XLEN.W), 0.U)
303  val pmpaddr2 = RegInit(UInt(XLEN.W), 0.U)
304  val pmpaddr3 = RegInit(UInt(XLEN.W), 0.U)
305
306  // Superviser-Level CSRs
307
308  // val sstatus = RegInit(UInt(XLEN.W), "h00000000".U)
309  val sstatusWmask = "hc6122".U
310  // Sstatus Write Mask
311  // -------------------------------------------------------
312  //    19           9   5     2
313  // 0  1100 0000 0001 0010 0010
314  // 0  c    0    1    2    2
315  // -------------------------------------------------------
316  val sstatusRmask = sstatusWmask | "h8000000300018000".U
317  // Sstatus Read Mask = (SSTATUS_WMASK | (0xf << 13) | (1ull << 63) | (3ull << 32))
318  val stvec = RegInit(UInt(XLEN.W), 0.U)
319  // val sie = RegInit(0.U(XLEN.W))
320  val sieMask = "h222".U & mideleg
321  val sipMask  = "h222".U & mideleg
322  val satp = if(EnbaleTlbDebug) RegInit(UInt(XLEN.W), "h8000000000087fbe".U) else RegInit(0.U(XLEN.W))
323  // val satp = RegInit(UInt(XLEN.W), "h8000000000087fbe".U) // only use for tlb naive debug
324  val satpMask = "h80000fffffffffff".U // disable asid, mode can only be 8 / 0
325  val sepc = RegInit(UInt(XLEN.W), 0.U)
326  val scause = RegInit(UInt(XLEN.W), 0.U)
327  val stval = Reg(UInt(XLEN.W))
328  val sscratch = RegInit(UInt(XLEN.W), 0.U)
329  val scounteren = RegInit(UInt(XLEN.W), 0.U)
330
331  // sbpctl
332  // Bits 0-7: {LOOP, RAS, SC, TAGE, BIM, BTB, uBTB}
333  val sbpctl = RegInit(UInt(XLEN.W), "h7f".U)
334  csrio.customCtrl.bp_ctrl.ubtb_enable := sbpctl(0)
335  csrio.customCtrl.bp_ctrl.btb_enable  := sbpctl(1)
336  csrio.customCtrl.bp_ctrl.bim_enable  := sbpctl(2)
337  csrio.customCtrl.bp_ctrl.tage_enable := sbpctl(3)
338  csrio.customCtrl.bp_ctrl.sc_enable   := sbpctl(4)
339  csrio.customCtrl.bp_ctrl.ras_enable  := sbpctl(5)
340  csrio.customCtrl.bp_ctrl.loop_enable := sbpctl(6)
341
342  // spfctl Bit 0: L1plusCache Prefetcher Enable
343  // spfctl Bit 1: L2Cache Prefetcher Enable
344  val spfctl = RegInit(UInt(XLEN.W), "h3".U)
345  csrio.customCtrl.l1plus_pf_enable := spfctl(0)
346  csrio.customCtrl.l2_pf_enable := spfctl(1)
347
348  // sdsid: Differentiated Services ID
349  val sdsid = RegInit(UInt(XLEN.W), 0.U)
350  csrio.customCtrl.dsid := sdsid
351
352  // slvpredctl: load violation predict settings
353  val slvpredctl = RegInit(UInt(XLEN.W), "h70".U) // default reset period: 2^17
354  csrio.customCtrl.lvpred_disable := slvpredctl(0)
355  csrio.customCtrl.no_spec_load := slvpredctl(1)
356  csrio.customCtrl.waittable_timeout := slvpredctl(8, 4)
357
358  // smblockctl: memory block configurations
359  // bits 0-3: store buffer flush threshold (default: 8 entries)
360  val smblockctl = RegInit(UInt(XLEN.W), "h7".U)
361  csrio.customCtrl.sbuffer_threshold := smblockctl(3, 0)
362
363  val tlbBundle = Wire(new TlbCsrBundle)
364  tlbBundle.satp := satp.asTypeOf(new SatpStruct)
365  csrio.tlb := tlbBundle
366
367  // User-Level CSRs
368  val uepc = Reg(UInt(XLEN.W))
369
370  // fcsr
371  class FcsrStruct extends Bundle {
372    val reserved = UInt((XLEN-3-5).W)
373    val frm = UInt(3.W)
374    val fflags = UInt(5.W)
375    assert(this.getWidth == XLEN)
376  }
377  val fcsr = RegInit(0.U(XLEN.W))
378  // set mstatus->sd and mstatus->fs when true
379  val csrw_dirty_fp_state = WireInit(false.B)
380
381  def frm_wfn(wdata: UInt): UInt = {
382    val fcsrOld = WireInit(fcsr.asTypeOf(new FcsrStruct))
383    csrw_dirty_fp_state := true.B
384    fcsrOld.frm := wdata(2,0)
385    fcsrOld.asUInt()
386  }
387  def frm_rfn(rdata: UInt): UInt = rdata(7,5)
388
389  def fflags_wfn(update: Boolean)(wdata: UInt): UInt = {
390    val fcsrOld = fcsr.asTypeOf(new FcsrStruct)
391    val fcsrNew = WireInit(fcsrOld)
392    csrw_dirty_fp_state := true.B
393    if (update) {
394      fcsrNew.fflags := wdata(4,0) | fcsrOld.fflags
395    } else {
396      fcsrNew.fflags := wdata(4,0)
397    }
398    fcsrNew.asUInt()
399  }
400  def fflags_rfn(rdata:UInt): UInt = rdata(4,0)
401
402  def fcsr_wfn(wdata: UInt): UInt = {
403    val fcsrOld = WireInit(fcsr.asTypeOf(new FcsrStruct))
404    csrw_dirty_fp_state := true.B
405    Cat(fcsrOld.reserved, wdata.asTypeOf(fcsrOld).frm, wdata.asTypeOf(fcsrOld).fflags)
406  }
407
408  val fcsrMapping = Map(
409    MaskedRegMap(Fflags, fcsr, wfn = fflags_wfn(update = false), rfn = fflags_rfn),
410    MaskedRegMap(Frm, fcsr, wfn = frm_wfn, rfn = frm_rfn),
411    MaskedRegMap(Fcsr, fcsr, wfn = fcsr_wfn)
412  )
413
414  // Atom LR/SC Control Bits
415  //  val setLr = WireInit(Bool(), false.B)
416  //  val setLrVal = WireInit(Bool(), false.B)
417  //  val setLrAddr = WireInit(UInt(AddrBits.W), DontCare) //TODO : need check
418  //  val lr = RegInit(Bool(), false.B)
419  //  val lrAddr = RegInit(UInt(AddrBits.W), 0.U)
420  //
421  //  when (setLr) {
422  //    lr := setLrVal
423  //    lrAddr := setLrAddr
424  //  }
425
426  // Hart Priviledge Mode
427  val priviledgeMode = RegInit(UInt(2.W), ModeM)
428
429  // Emu perfcnt
430  val hasEmuPerfCnt = !env.FPGAPlatform
431  val nrEmuPerfCnts = if (hasEmuPerfCnt) 0x80 else 0x3
432
433  val emuPerfCnts    = List.fill(nrEmuPerfCnts)(RegInit(0.U(XLEN.W)))
434  val emuPerfCntCond = List.fill(nrEmuPerfCnts)(WireInit(false.B))
435  (emuPerfCnts zip emuPerfCntCond).map { case (c, e) => when (e) { c := c + 1.U } }
436
437  val emuPerfCntsLoMapping = (0 until nrEmuPerfCnts).map(i => MaskedRegMap(0x1000 + i, emuPerfCnts(i)))
438  val emuPerfCntsHiMapping = (0 until nrEmuPerfCnts).map(i => MaskedRegMap(0x1080 + i, emuPerfCnts(i)(63, 32)))
439  println(s"CSR: hasEmuPerfCnt:${hasEmuPerfCnt}")
440
441  // Perf Counter
442  val nrPerfCnts = 29  // 3...31
443  val perfCnts   = List.fill(nrPerfCnts)(RegInit(0.U(XLEN.W)))
444  val perfEvents = List.fill(nrPerfCnts)(RegInit(0.U(XLEN.W)))
445  val mcountinhibit = RegInit(0.U(XLEN.W))
446  val mcycle = RegInit(0.U(XLEN.W))
447  mcycle := mcycle + 1.U
448  val minstret = RegInit(0.U(XLEN.W))
449  minstret := minstret + RegNext(csrio.perf.retiredInstr)
450
451  // CSR reg map
452  val basicPrivMapping = Map(
453
454    //--- User Trap Setup ---
455    // MaskedRegMap(Ustatus, ustatus),
456    // MaskedRegMap(Uie, uie, 0.U, MaskedRegMap.Unwritable),
457    // MaskedRegMap(Utvec, utvec),
458
459    //--- User Trap Handling ---
460    // MaskedRegMap(Uscratch, uscratch),
461    // MaskedRegMap(Uepc, uepc),
462    // MaskedRegMap(Ucause, ucause),
463    // MaskedRegMap(Utval, utval),
464    // MaskedRegMap(Uip, uip),
465
466    //--- User Counter/Timers ---
467    // MaskedRegMap(Cycle, cycle),
468    // MaskedRegMap(Time, time),
469    // MaskedRegMap(Instret, instret),
470
471    //--- Supervisor Trap Setup ---
472    MaskedRegMap(Sstatus, mstatus, sstatusWmask, mstatusUpdateSideEffect, sstatusRmask),
473    // MaskedRegMap(Sedeleg, Sedeleg),
474    // MaskedRegMap(Sideleg, Sideleg),
475    MaskedRegMap(Sie, mie, sieMask, MaskedRegMap.NoSideEffect, sieMask),
476    MaskedRegMap(Stvec, stvec),
477    MaskedRegMap(Scounteren, scounteren),
478
479    //--- Supervisor Trap Handling ---
480    MaskedRegMap(Sscratch, sscratch),
481    MaskedRegMap(Sepc, sepc),
482    MaskedRegMap(Scause, scause),
483    MaskedRegMap(Stval, stval),
484    MaskedRegMap(Sip, mip.asUInt, sipMask, MaskedRegMap.Unwritable, sipMask),
485
486    //--- Supervisor Protection and Translation ---
487    MaskedRegMap(Satp, satp, satpMask, MaskedRegMap.NoSideEffect, satpMask),
488
489    //--- Supervisor Custom Read/Write Registers
490    MaskedRegMap(Sbpctl, sbpctl),
491    MaskedRegMap(Spfctl, spfctl),
492    MaskedRegMap(Sdsid, sdsid),
493    MaskedRegMap(Slvpredctl, slvpredctl),
494    MaskedRegMap(Smblockctl, smblockctl),
495
496    //--- Machine Information Registers ---
497    MaskedRegMap(Mvendorid, mvendorid, 0.U, MaskedRegMap.Unwritable),
498    MaskedRegMap(Marchid, marchid, 0.U, MaskedRegMap.Unwritable),
499    MaskedRegMap(Mimpid, mimpid, 0.U, MaskedRegMap.Unwritable),
500    MaskedRegMap(Mhartid, mhartid, 0.U, MaskedRegMap.Unwritable),
501
502    //--- Machine Trap Setup ---
503    MaskedRegMap(Mstatus, mstatus, mstatusMask, mstatusUpdateSideEffect, mstatusMask),
504    MaskedRegMap(Misa, misa), // now MXL, EXT is not changeable
505    MaskedRegMap(Medeleg, medeleg, "hf3ff".U),
506    MaskedRegMap(Mideleg, mideleg, "h222".U),
507    MaskedRegMap(Mie, mie),
508    MaskedRegMap(Mtvec, mtvec),
509    MaskedRegMap(Mcounteren, mcounteren),
510
511    //--- Machine Trap Handling ---
512    MaskedRegMap(Mscratch, mscratch),
513    MaskedRegMap(Mepc, mepc),
514    MaskedRegMap(Mcause, mcause),
515    MaskedRegMap(Mtval, mtval),
516    MaskedRegMap(Mip, mip.asUInt, 0.U, MaskedRegMap.Unwritable),
517  )
518
519  // PMP is unimplemented yet
520  val pmpMapping = Map(
521    MaskedRegMap(Pmpcfg0, pmpcfg0),
522    MaskedRegMap(Pmpcfg1, pmpcfg1),
523    MaskedRegMap(Pmpcfg2, pmpcfg2),
524    MaskedRegMap(Pmpcfg3, pmpcfg3),
525    MaskedRegMap(PmpaddrBase + 0, pmpaddr0),
526    MaskedRegMap(PmpaddrBase + 1, pmpaddr1),
527    MaskedRegMap(PmpaddrBase + 2, pmpaddr2),
528    MaskedRegMap(PmpaddrBase + 3, pmpaddr3)
529  )
530
531  var perfCntMapping = Map(
532    MaskedRegMap(Mcountinhibit, mcountinhibit),
533    MaskedRegMap(Mcycle, mcycle),
534    MaskedRegMap(Minstret, minstret),
535  )
536  val MhpmcounterStart = Mhpmcounter3
537  val MhpmeventStart   = Mhpmevent3
538  for (i <- 0 until nrPerfCnts) {
539    perfCntMapping += MaskedRegMap(MhpmcounterStart + i, perfCnts(i))
540    perfCntMapping += MaskedRegMap(MhpmeventStart + i, perfEvents(i))
541  }
542
543  val mapping = basicPrivMapping ++
544                perfCntMapping ++
545                pmpMapping ++
546                emuPerfCntsLoMapping ++
547                (if (XLEN == 32) emuPerfCntsHiMapping else Nil) ++
548                (if (HasFPU) fcsrMapping else Nil)
549
550  val addr = src2(11, 0)
551  val csri = ZeroExt(src2(16, 12), XLEN)
552  val rdata = Wire(UInt(XLEN.W))
553  val wdata = LookupTree(func, List(
554    CSROpType.wrt  -> src1,
555    CSROpType.set  -> (rdata | src1),
556    CSROpType.clr  -> (rdata & (~src1).asUInt()),
557    CSROpType.wrti -> csri,
558    CSROpType.seti -> (rdata | csri),
559    CSROpType.clri -> (rdata & (~csri).asUInt())
560  ))
561
562  val addrInPerfCnt = (addr >= Mcycle.U) && (addr <= Mhpmcounter31.U)
563  csrio.isPerfCnt := addrInPerfCnt
564
565  // satp wen check
566  val satpLegalMode = (wdata.asTypeOf(new SatpStruct).mode===0.U) || (wdata.asTypeOf(new SatpStruct).mode===8.U)
567
568  // general CSR wen check
569  val wen = valid && func =/= CSROpType.jmp && (addr=/=Satp.U || satpLegalMode)
570  val modePermitted = csrAccessPermissionCheck(addr, false.B, priviledgeMode)
571  val perfcntPermitted = perfcntPermissionCheck(addr, priviledgeMode, mcounteren, scounteren)
572  val permitted = Mux(addrInPerfCnt, perfcntPermitted, modePermitted)
573  // Writeable check is ingored.
574  // Currently, write to illegal csr addr will be ignored
575  MaskedRegMap.generate(mapping, addr, rdata, wen && permitted, wdata)
576  io.out.bits.data := rdata
577  io.out.bits.uop := io.in.bits.uop
578  io.out.bits.uop.cf := cfOut
579  io.out.bits.uop.ctrl.flushPipe := flushPipe
580
581  // Fix Mip/Sip write
582  val fixMapping = Map(
583    MaskedRegMap(Mip, mipReg.asUInt, mipFixMask),
584    MaskedRegMap(Sip, mipReg.asUInt, sipMask, MaskedRegMap.NoSideEffect, sipMask)
585  )
586  val rdataDummy = Wire(UInt(XLEN.W))
587  MaskedRegMap.generate(fixMapping, addr, rdataDummy, wen, wdata)
588
589  when (csrio.fpu.fflags.valid) {
590    fcsr := fflags_wfn(update = true)(csrio.fpu.fflags.bits)
591  }
592  // set fs and sd in mstatus
593  when (csrw_dirty_fp_state || csrio.fpu.dirty_fs) {
594    val mstatusNew = WireInit(mstatus.asTypeOf(new MstatusStruct))
595    mstatusNew.fs := "b11".U
596    mstatusNew.sd := true.B
597    mstatus := mstatusNew.asUInt()
598  }
599  csrio.fpu.frm := fcsr.asTypeOf(new FcsrStruct).frm
600
601  // CSR inst decode
602  val isEbreak = addr === privEbreak && func === CSROpType.jmp
603  val isEcall  = addr === privEcall  && func === CSROpType.jmp
604  val isMret   = addr === privMret   && func === CSROpType.jmp
605  val isSret   = addr === privSret   && func === CSROpType.jmp
606  val isUret   = addr === privUret   && func === CSROpType.jmp
607
608  XSDebug(wen, "csr write: pc %x addr %x rdata %x wdata %x func %x\n", cfIn.pc, addr, rdata, wdata, func)
609  XSDebug(wen, "pc %x mstatus %x mideleg %x medeleg %x mode %x\n", cfIn.pc, mstatus, mideleg , medeleg, priviledgeMode)
610
611  // Illegal priviledged operation list
612  val illegalSModeSret = valid && isSret && priviledgeMode === ModeS && mstatusStruct.tsr.asBool
613
614  // Illegal priviledged instruction check
615  val isIllegalAddr = MaskedRegMap.isIllegalAddr(mapping, addr)
616  val isIllegalAccess = !permitted
617  val isIllegalPrivOp = illegalSModeSret
618
619  // def MMUPermissionCheck(ptev: Bool, pteu: Bool): Bool = ptev && !(priviledgeMode === ModeU && !pteu) && !(priviledgeMode === ModeS && pteu && mstatusStruct.sum.asBool)
620  // def MMUPermissionCheckLoad(ptev: Bool, pteu: Bool): Bool = ptev && !(priviledgeMode === ModeU && !pteu) && !(priviledgeMode === ModeS && pteu && mstatusStruct.sum.asBool) && (pter || (mstatusStruct.mxr && ptex))
621  // imem
622  // val imemPtev = true.B
623  // val imemPteu = true.B
624  // val imemPtex = true.B
625  // val imemReq = true.B
626  // val imemPermissionCheckPassed = MMUPermissionCheck(imemPtev, imemPteu)
627  // val hasInstrPageFault = imemReq && !(imemPermissionCheckPassed && imemPtex)
628  // assert(!hasInstrPageFault)
629
630  // dmem
631  // val dmemPtev = true.B
632  // val dmemPteu = true.B
633  // val dmemReq = true.B
634  // val dmemPermissionCheckPassed = MMUPermissionCheck(dmemPtev, dmemPteu)
635  // val dmemIsStore = true.B
636
637  // val hasLoadPageFault  = dmemReq && !dmemIsStore && !(dmemPermissionCheckPassed)
638  // val hasStorePageFault = dmemReq &&  dmemIsStore && !(dmemPermissionCheckPassed)
639  // assert(!hasLoadPageFault)
640  // assert(!hasStorePageFault)
641
642  //TODO: Havn't test if io.dmemMMU.priviledgeMode is correct yet
643  tlbBundle.priv.mxr   := mstatusStruct.mxr.asBool
644  tlbBundle.priv.sum   := mstatusStruct.sum.asBool
645  tlbBundle.priv.imode := priviledgeMode
646  tlbBundle.priv.dmode := Mux(mstatusStruct.mprv.asBool, mstatusStruct.mpp, priviledgeMode)
647
648  // Branch control
649  val retTarget = Wire(UInt(VAddrBits.W))
650  val resetSatp = addr === Satp.U && wen // write to satp will cause the pipeline be flushed
651  flushPipe := resetSatp || (valid && func === CSROpType.jmp && !isEcall)
652
653  retTarget := DontCare
654  // val illegalEret = TODO
655
656  when (valid && isMret) {
657    val mstatusOld = WireInit(mstatus.asTypeOf(new MstatusStruct))
658    val mstatusNew = WireInit(mstatus.asTypeOf(new MstatusStruct))
659    mstatusNew.ie.m := mstatusOld.pie.m
660    priviledgeMode := mstatusOld.mpp
661    mstatusNew.pie.m := true.B
662    mstatusNew.mpp := ModeU
663    mstatusNew.mprv := 0.U
664    mstatus := mstatusNew.asUInt
665    // lr := false.B
666    retTarget := mepc(VAddrBits-1, 0)
667  }
668
669  when (valid && isSret && !illegalSModeSret) {
670    val mstatusOld = WireInit(mstatus.asTypeOf(new MstatusStruct))
671    val mstatusNew = WireInit(mstatus.asTypeOf(new MstatusStruct))
672    mstatusNew.ie.s := mstatusOld.pie.s
673    priviledgeMode := Cat(0.U(1.W), mstatusOld.spp)
674    mstatusNew.pie.s := true.B
675    mstatusNew.spp := ModeU
676    mstatus := mstatusNew.asUInt
677    mstatusNew.mprv := 0.U
678    // lr := false.B
679    retTarget := sepc(VAddrBits-1, 0)
680  }
681
682  when (valid && isUret) {
683    val mstatusOld = WireInit(mstatus.asTypeOf(new MstatusStruct))
684    val mstatusNew = WireInit(mstatus.asTypeOf(new MstatusStruct))
685    // mstatusNew.mpp.m := ModeU //TODO: add mode U
686    mstatusNew.ie.u := mstatusOld.pie.u
687    priviledgeMode := ModeU
688    mstatusNew.pie.u := true.B
689    mstatus := mstatusNew.asUInt
690    retTarget := uepc(VAddrBits-1, 0)
691  }
692
693  io.in.ready := true.B
694  io.out.valid := valid
695
696  val csrExceptionVec = WireInit(cfIn.exceptionVec)
697  csrExceptionVec(breakPoint) := io.in.valid && isEbreak
698  csrExceptionVec(ecallM) := priviledgeMode === ModeM && io.in.valid && isEcall
699  csrExceptionVec(ecallS) := priviledgeMode === ModeS && io.in.valid && isEcall
700  csrExceptionVec(ecallU) := priviledgeMode === ModeU && io.in.valid && isEcall
701  // Trigger an illegal instr exception when:
702  // * unimplemented csr is being read/written
703  // * csr access is illegal
704  csrExceptionVec(illegalInstr) := (isIllegalAddr || isIllegalAccess) && wen
705  cfOut.exceptionVec := csrExceptionVec
706
707  /**
708    * Exception and Intr
709    */
710  val ideleg =  (mideleg & mip.asUInt)
711  def priviledgedEnableDetect(x: Bool): Bool = Mux(x, ((priviledgeMode === ModeS) && mstatusStruct.ie.s) || (priviledgeMode < ModeS),
712    ((priviledgeMode === ModeM) && mstatusStruct.ie.m) || (priviledgeMode < ModeM))
713
714  // send interrupt information to ROQ
715  val intrVecEnable = Wire(Vec(12, Bool()))
716  intrVecEnable.zip(ideleg.asBools).map{case(x,y) => x := priviledgedEnableDetect(y)}
717  val intrVec = mie(11,0) & mip.asUInt & intrVecEnable.asUInt
718  val intrBitSet = intrVec.orR()
719  csrio.interrupt := intrBitSet
720  mipWire.t.m := csrio.externalInterrupt.mtip
721  mipWire.s.m := csrio.externalInterrupt.msip
722  mipWire.e.m := csrio.externalInterrupt.meip
723
724  // interrupts
725  val intrNO = IntPriority.foldRight(0.U)((i: Int, sum: UInt) => Mux(intrVec(i), i.U, sum))
726  val raiseIntr = csrio.exception.valid && csrio.exception.bits.isInterrupt
727  XSDebug(raiseIntr, "interrupt: pc=0x%x, %d\n", csrio.exception.bits.uop.cf.pc, intrNO)
728
729  // exceptions
730  val raiseException = csrio.exception.valid && !csrio.exception.bits.isInterrupt
731  val hasInstrPageFault = csrio.exception.bits.uop.cf.exceptionVec(instrPageFault) && raiseException
732  val hasLoadPageFault = csrio.exception.bits.uop.cf.exceptionVec(loadPageFault) && raiseException
733  val hasStorePageFault = csrio.exception.bits.uop.cf.exceptionVec(storePageFault) && raiseException
734  val hasStoreAddrMisaligned = csrio.exception.bits.uop.cf.exceptionVec(storeAddrMisaligned) && raiseException
735  val hasLoadAddrMisaligned = csrio.exception.bits.uop.cf.exceptionVec(loadAddrMisaligned) && raiseException
736  val hasInstrAccessFault = csrio.exception.bits.uop.cf.exceptionVec(instrAccessFault) && raiseException
737  val hasLoadAccessFault = csrio.exception.bits.uop.cf.exceptionVec(loadAccessFault) && raiseException
738  val hasStoreAccessFault = csrio.exception.bits.uop.cf.exceptionVec(storeAccessFault) && raiseException
739
740  val raiseExceptionVec = csrio.exception.bits.uop.cf.exceptionVec
741  val exceptionNO = ExcPriority.foldRight(0.U)((i: Int, sum: UInt) => Mux(raiseExceptionVec(i), i.U, sum))
742  val causeNO = (raiseIntr << (XLEN-1)).asUInt() | Mux(raiseIntr, intrNO, exceptionNO)
743
744  val raiseExceptionIntr = csrio.exception.valid
745  XSDebug(raiseExceptionIntr, "int/exc: pc %x int (%d):%x exc: (%d):%x\n",
746    csrio.exception.bits.uop.cf.pc, intrNO, intrVec, exceptionNO, raiseExceptionVec.asUInt
747  )
748  XSDebug(raiseExceptionIntr,
749    "pc %x mstatus %x mideleg %x medeleg %x mode %x\n",
750    csrio.exception.bits.uop.cf.pc,
751    mstatus,
752    mideleg,
753    medeleg,
754    priviledgeMode
755  )
756
757  // mtval write logic
758  val memExceptionAddr = SignExt(csrio.memExceptionVAddr, XLEN)
759  when (hasInstrPageFault || hasLoadPageFault || hasStorePageFault) {
760    val tval = Mux(
761      hasInstrPageFault,
762      Mux(
763        csrio.exception.bits.uop.cf.crossPageIPFFix,
764        SignExt(csrio.exception.bits.uop.cf.pc + 2.U, XLEN),
765        SignExt(csrio.exception.bits.uop.cf.pc, XLEN)
766      ),
767      memExceptionAddr
768    )
769    when (priviledgeMode === ModeM) {
770      mtval := tval
771    }.otherwise {
772      stval := tval
773    }
774  }
775
776  when (hasLoadAddrMisaligned || hasStoreAddrMisaligned) {
777    mtval := memExceptionAddr
778  }
779
780  val deleg = Mux(raiseIntr, mideleg , medeleg)
781  // val delegS = ((deleg & (1 << (causeNO & 0xf))) != 0) && (priviledgeMode < ModeM);
782  val delegS = deleg(causeNO(3,0)) && (priviledgeMode < ModeM)
783  val tvalWen = !(hasInstrPageFault || hasLoadPageFault || hasStorePageFault || hasLoadAddrMisaligned || hasStoreAddrMisaligned) || raiseIntr // TODO: need check
784  val isXRet = io.in.valid && func === CSROpType.jmp && !isEcall
785
786  // ctrl block will use theses later for flush
787  val isXRetFlag = RegInit(false.B)
788  val retTargetReg = Reg(retTarget.cloneType)
789  when (io.flushIn) {
790    isXRetFlag := false.B
791  }.elsewhen (isXRet) {
792    isXRetFlag := true.B
793    retTargetReg := retTarget
794  }
795  csrio.isXRet := isXRetFlag
796  csrio.trapTarget := Mux(isXRetFlag,
797    retTargetReg,
798    Mux(delegS, stvec, mtvec)(VAddrBits-1, 0)
799  )
800
801  when (raiseExceptionIntr) {
802    val mstatusOld = WireInit(mstatus.asTypeOf(new MstatusStruct))
803    val mstatusNew = WireInit(mstatus.asTypeOf(new MstatusStruct))
804
805    when (delegS) {
806      scause := causeNO
807      sepc := SignExt(csrio.exception.bits.uop.cf.pc, XLEN)
808      mstatusNew.spp := priviledgeMode
809      mstatusNew.pie.s := mstatusOld.ie.s
810      mstatusNew.ie.s := false.B
811      priviledgeMode := ModeS
812      when (tvalWen) { stval := 0.U }
813    }.otherwise {
814      mcause := causeNO
815      mepc := SignExt(csrio.exception.bits.uop.cf.pc, XLEN)
816      mstatusNew.mpp := priviledgeMode
817      mstatusNew.pie.m := mstatusOld.ie.m
818      mstatusNew.ie.m := false.B
819      priviledgeMode := ModeM
820      when (tvalWen) { mtval := 0.U }
821    }
822
823    mstatus := mstatusNew.asUInt
824  }
825
826  XSDebug(raiseExceptionIntr && delegS, "sepc is writen!!! pc:%x\n", cfIn.pc)
827
828  def readWithScala(addr: Int): UInt = mapping(addr)._1
829
830  val difftestIntrNO = Mux(raiseIntr, causeNO, 0.U)
831
832  if (!env.FPGAPlatform) {
833    difftestIO.intrNO := RegNext(difftestIntrNO)
834    difftestIO.cause := RegNext(Mux(csrio.exception.valid, causeNO, 0.U))
835    difftestIO.priviledgeMode := priviledgeMode
836    difftestIO.mstatus := mstatus
837    difftestIO.sstatus := mstatus & sstatusRmask
838    difftestIO.mepc := mepc
839    difftestIO.sepc := sepc
840    difftestIO.mtval:= mtval
841    difftestIO.stval:= stval
842    difftestIO.mtvec := mtvec
843    difftestIO.stvec := stvec
844    difftestIO.mcause := mcause
845    difftestIO.scause := scause
846    difftestIO.satp := satp
847    difftestIO.mip := mipReg
848    difftestIO.mie := mie
849    difftestIO.mscratch := mscratch
850    difftestIO.sscratch := sscratch
851    difftestIO.mideleg := mideleg
852    difftestIO.medeleg := medeleg
853  }
854}
855