xref: /XiangShan/src/main/scala/xiangshan/backend/fu/NewCSR/Unprivileged.scala (revision fc74c6e4974d29a63a39a3a1715a369e1236fa06)
1package xiangshan.backend.fu.NewCSR
2
3import chisel3._
4import chisel3.util._
5import freechips.rocketchip.rocket.CSRs
6import utility.GatedValidRegNext
7import xiangshan.backend.fu.NewCSR.CSRDefines.{CSRROField => RO, CSRRWField => RW, CSRWARLField => WARL}
8import xiangshan.backend.fu.NewCSR.CSRFunc._
9import xiangshan.backend.fu.vector.Bundles._
10import xiangshan.backend.fu.NewCSR.CSRConfig._
11import xiangshan.backend.fu.fpu.Bundles.{Fflags, Frm}
12import xiangshan.backend.fu.NewCSR.CSREnumTypeImplicitCast._
13
14import scala.collection.immutable.SeqMap
15
16trait Unprivileged { self: NewCSR with MachineLevel with SupervisorLevel =>
17
18  val fcsr = Module(new CSRModule("Fcsr", new CSRBundle {
19    val NX = WARL(0, wNoFilter)
20    val UF = WARL(1, wNoFilter)
21    val OF = WARL(2, wNoFilter)
22    val DZ = WARL(3, wNoFilter)
23    val NV = WARL(4, wNoFilter)
24    val FRM = WARL(7, 5, wNoFilter).withReset(0.U)
25  }) with HasRobCommitBundle {
26    val wAliasFflags = IO(Input(new CSRAddrWriteBundle(new CSRFFlagsBundle)))
27    val wAliasFfm = IO(Input(new CSRAddrWriteBundle(new CSRFrmBundle)))
28    val fflags = IO(Output(Fflags()))
29    val frm = IO(Output(Frm()))
30    val fflagsRdata = IO(Output(Fflags()))
31    val frmRdata = IO(Output(Frm()))
32
33    for (wAlias <- Seq(wAliasFflags, wAliasFfm)) {
34      for ((name, field) <- wAlias.wdataFields.elements) {
35        reg.elements(name).asInstanceOf[CSREnumType].addOtherUpdate(
36          wAlias.wen && field.asInstanceOf[CSREnumType].isLegal,
37          field.asInstanceOf[CSREnumType]
38        )
39      }
40    }
41
42    // write connection
43    reconnectReg()
44
45    when (robCommit.fflags.valid) {
46      reg.NX := robCommit.fflags.bits(0) || reg.NX
47      reg.UF := robCommit.fflags.bits(1) || reg.UF
48      reg.OF := robCommit.fflags.bits(2) || reg.OF
49      reg.DZ := robCommit.fflags.bits(3) || reg.DZ
50      reg.NV := robCommit.fflags.bits(4) || reg.NV
51    }
52
53    // read connection
54    fflags := reg.asUInt(4, 0)
55    frm := reg.FRM.asUInt
56
57    fflagsRdata := fflags.asUInt
58    frmRdata := frm.asUInt
59  }).setAddr(CSRs.fcsr)
60
61  // vec
62  val vstart = Module(new CSRModule("Vstart", new CSRBundle {
63    // vstart is not a WARL CSR.
64    // Since we need to judge whether flush pipe by vstart being not 0 in DecodeStage, vstart must be initialized to some value at reset.
65    val vstart = RW(VlWidth - 2, 0).withReset(0.U) // hold [0, 128)
66  }) with HasRobCommitBundle {
67    // Todo make The use of vstart values greater than the largest element index for the current SEW setting is reserved.
68    // Not trap
69    when (wen) {
70      reg.vstart := this.w.wdata(VlWidth - 2, 0)
71    }.elsewhen (robCommit.vsDirty && !robCommit.vstart.valid) {
72      reg.vstart := 0.U
73    }.elsewhen (robCommit.vstart.valid) {
74      reg.vstart := robCommit.vstart.bits
75    }.otherwise {
76      reg := reg
77    }
78  })
79    .setAddr(CSRs.vstart)
80
81  val vcsr = Module(new CSRModule("Vcsr", new CSRBundle {
82    val VXSAT = RW(   0)
83    val VXRM  = RW(2, 1)
84  }) with HasRobCommitBundle {
85    val wAliasVxsat = IO(Input(new CSRAddrWriteBundle(new CSRBundle {
86      val VXSAT = RW(0)
87    })))
88    val wAlisaVxrm = IO(Input(new CSRAddrWriteBundle(new CSRBundle {
89      val VXRM = RW(1, 0)
90    })))
91    val vxsat = IO(Output(Vxsat()))
92    val vxrm  = IO(Output(Vxrm()))
93
94    for (wAlias <- Seq(wAliasVxsat, wAlisaVxrm)) {
95      for ((name, field) <- wAlias.wdataFields.elements) {
96        reg.elements(name).asInstanceOf[CSREnumType].addOtherUpdate(
97          wAlias.wen && field.asInstanceOf[CSREnumType].isLegal,
98          field.asInstanceOf[CSREnumType]
99        )
100      }
101    }
102
103    // write connection
104    reconnectReg()
105
106    when(robCommit.vxsat.valid) {
107      reg.VXSAT := reg.VXSAT.asBool || robCommit.vxsat.bits.asBool
108    }
109
110    // read connection
111    vxsat := reg.VXSAT.asUInt
112    vxrm  := reg.VXRM.asUInt
113  }).setAddr(CSRs.vcsr)
114
115  val vl = Module(new CSRModule("Vl", new CSRBundle {
116    val VL = RO(VlWidth - 1, 0).withReset(0.U)
117  }))
118    .setAddr(CSRs.vl)
119
120  val vtype = Module(new CSRModule("Vtype", new CSRVTypeBundle) with HasRobCommitBundle {
121    when(robCommit.vtype.valid) {
122      reg := robCommit.vtype.bits
123    }
124  })
125    .setAddr(CSRs.vtype)
126
127  val vlenb = Module(new CSRModule("Vlenb", new CSRBundle {
128    val VLENB = VlenbField(63, 0).withReset(VlenbField.init)
129  }))
130    .setAddr(CSRs.vlenb)
131
132  val cycle = Module(new CSRModule("cycle", new CSRBundle {
133    val cycle = RO(63, 0)
134  }) with HasMHPMSink {
135    regOut.cycle := mHPM.cycle
136  })
137    .setAddr(CSRs.cycle)
138
139  val time = Module(new CSRModule("time", new CSRBundle {
140    val time = RO(63, 0)
141  }) with HasMHPMSink {
142    val updated = IO(Output(Bool()))
143    val stime  = IO(Output(UInt(64.W)))
144    val vstime = IO(Output(UInt(64.W)))
145
146    val stimeTmp  = mHPM.time.bits
147    val vstimeTmp = mHPM.time.bits + htimedelta
148
149    when (mHPM.time.valid) {
150      reg.time := Mux(v, vstimeTmp, stimeTmp)
151    }
152
153    updated := GatedValidRegNext(mHPM.time.valid)
154    stime  := stimeTmp
155    vstime := vstimeTmp
156  })
157    .setAddr(CSRs.time)
158
159  val instret = Module(new CSRModule("instret", new CSRBundle {
160    val instret = RO(63, 0)
161  }) with HasMHPMSink {
162    regOut.instret := mHPM.instret
163  })
164    .setAddr(CSRs.instret)
165
166  val hpmcounters: Seq[CSRModule[_]] = (3 to 0x1F).map(num =>
167    Module(new CSRModule(s"Hpmcounter$num", new CSRBundle {
168      val hpmcounter = RO(63, 0).withReset(0.U)
169    }) with HasMHPMSink {
170      regOut.hpmcounter := mHPM.hpmcounters(num - 3)
171    }).setAddr(CSRs.cycle + num)
172  )
173
174  val unprivilegedCSRMap: SeqMap[Int, (CSRAddrWriteBundle[_], UInt)] = SeqMap(
175    CSRs.fflags -> (fcsr.wAliasFflags -> fcsr.fflagsRdata),
176    CSRs.frm    -> (fcsr.wAliasFfm    -> fcsr.frmRdata),
177    CSRs.fcsr   -> (fcsr.w            -> fcsr.rdata),
178    CSRs.vstart -> (vstart.w          -> vstart.rdata),
179    CSRs.vxsat  -> (vcsr.wAliasVxsat  -> vcsr.vxsat),
180    CSRs.vxrm   -> (vcsr.wAlisaVxrm   -> vcsr.vxrm),
181    CSRs.vcsr   -> (vcsr.w            -> vcsr.rdata),
182    CSRs.vl     -> (vl.w              -> vl.rdata),
183    CSRs.vtype  -> (vtype.w           -> vtype.rdata),
184    CSRs.vlenb  -> (vlenb.w           -> vlenb.rdata),
185    CSRs.cycle  -> (cycle.w           -> cycle.rdata),
186    CSRs.time   -> (time.w            -> time.rdata),
187    CSRs.instret -> (instret.w        -> instret.rdata),
188  ) ++ hpmcounters.map(counter => (counter.addr -> (counter.w -> counter.rdata)))
189
190  val unprivilegedCSRMods: Seq[CSRModule[_]] = Seq(
191    fcsr,
192    vcsr,
193    vstart,
194    vl,
195    vtype,
196    vlenb,
197    cycle,
198    time,
199    instret,
200  ) ++ hpmcounters
201
202  val unprivilegedCSROutMap: SeqMap[Int, UInt] = SeqMap(
203    CSRs.fflags  -> fcsr.fflags.asUInt,
204    CSRs.frm     -> fcsr.frm.asUInt,
205    CSRs.fcsr    -> fcsr.rdata.asUInt,
206    CSRs.vstart  -> vstart.rdata.asUInt,
207    CSRs.vxsat   -> vcsr.vxsat.asUInt,
208    CSRs.vxrm    -> vcsr.vxrm.asUInt,
209    CSRs.vcsr    -> vcsr.rdata.asUInt,
210    CSRs.vl      -> vl.rdata.asUInt,
211    CSRs.vtype   -> vtype.rdata.asUInt,
212    CSRs.vlenb   -> vlenb.rdata.asUInt,
213    CSRs.cycle   -> cycle.rdata,
214    CSRs.time    -> time.rdata,
215    CSRs.instret -> instret.rdata,
216  ) ++ hpmcounters.map(counter => (counter.addr -> counter.rdata))
217}
218
219class CSRVTypeBundle extends CSRBundle {
220  // vtype's vill is initialized to 1, when executing vector instructions
221  // which depend on vtype, will raise illegal instruction exception
222  val VILL  = RO(  63).withReset(1.U)
223  val VMA   = RO(   7).withReset(0.U)
224  val VTA   = RO(   6).withReset(0.U)
225  val VSEW  = RO(5, 3).withReset(0.U)
226  val VLMUL = RO(2, 0).withReset(0.U)
227}
228
229class CSRFrmBundle extends CSRBundle {
230  val FRM = WARL(2, 0, wNoFilter)
231}
232
233class CSRFFlagsBundle extends CSRBundle {
234  val NX = WARL(0, wNoFilter)
235  val UF = WARL(1, wNoFilter)
236  val OF = WARL(2, wNoFilter)
237  val DZ = WARL(3, wNoFilter)
238  val NV = WARL(4, wNoFilter)
239}
240
241object VlenbField extends CSREnum with ROApply {
242  val init = Value((VLEN / 8).U)
243}
244
245trait HasMHPMSink { self: CSRModule[_] =>
246  val mHPM = IO(Input(new Bundle {
247    val cycle   = UInt(64.W)
248    // ValidIO is used to update time reg
249    val time    = ValidIO(UInt(64.W))
250    val instret = UInt(64.W)
251    val hpmcounters = Vec(perfCntNum, UInt(XLEN.W))
252  }))
253  val v = IO(Input(Bool()))
254  val htimedelta = IO(Input(UInt(64.W)))
255}
256