xref: /XiangShan/src/main/scala/device/AXI4Plic.scala (revision c6d439803a044ea209139672b25e35fe8d7f4aa0)
1*c6d43980SLemover/***************************************************************************************
2*c6d43980SLemover* Copyright (c) 2020-2021 Institute of Computing Technology, Chinese Academy of Sciences
3*c6d43980SLemover*
4*c6d43980SLemover* XiangShan is licensed under Mulan PSL v2.
5*c6d43980SLemover* You can use this software according to the terms and conditions of the Mulan PSL v2.
6*c6d43980SLemover* You may obtain a copy of Mulan PSL v2 at:
7*c6d43980SLemover*          http://license.coscl.org.cn/MulanPSL2
8*c6d43980SLemover*
9*c6d43980SLemover* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
10*c6d43980SLemover* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
11*c6d43980SLemover* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
12*c6d43980SLemover*
13*c6d43980SLemover* See the Mulan PSL v2 for more details.
14*c6d43980SLemover***************************************************************************************/
15*c6d43980SLemover
16c699b48aSwangkaifanpackage device
17c699b48aSwangkaifan
18c699b48aSwangkaifanimport chisel3._
19c699b48aSwangkaifanimport chisel3.util._
20c699b48aSwangkaifanimport chipsalliance.rocketchip.config._
21c699b48aSwangkaifanimport freechips.rocketchip.diplomacy._
22c699b48aSwangkaifanimport utils.MaskExpand
23c699b48aSwangkaifanimport utils.{HasTLDump, XSDebug, RegMap}
24c699b48aSwangkaifan
25c699b48aSwangkaifan/*  base + 0x000000: Reserved (interrupt source 0 does not exist)
26c699b48aSwangkaifan    base + 0x000004: Interrupt source 1 priority
27c699b48aSwangkaifan    base + 0x000008: Interrupt source 2 priority
28c699b48aSwangkaifan    ...
29c699b48aSwangkaifan    base + 0x000FFC: Interrupt source 1023 priority
30c699b48aSwangkaifan    base + 0x001000: Interrupt Pending bit 0-31
31c699b48aSwangkaifan    base + 0x00107C: Interrupt Pending bit 992-1023
32c699b48aSwangkaifan    ...
33c699b48aSwangkaifan    base + 0x002000: Enable bits for sources 0-31 on context 0
34c699b48aSwangkaifan    base + 0x002004: Enable bits for sources 32-63 on context 0
35c699b48aSwangkaifan    ...
36c699b48aSwangkaifan    base + 0x00207F: Enable bits for sources 992-1023 on context 0
37c699b48aSwangkaifan    base + 0x002080: Enable bits for sources 0-31 on context 1
38c699b48aSwangkaifan    base + 0x002084: Enable bits for sources 32-63 on context 1
39c699b48aSwangkaifan    ...
40c699b48aSwangkaifan    base + 0x0020FF: Enable bits for sources 992-1023 on context 1
41c699b48aSwangkaifan    base + 0x002100: Enable bits for sources 0-31 on context 2
42c699b48aSwangkaifan    base + 0x002104: Enable bits for sources 32-63 on context 2
43c699b48aSwangkaifan    ...
44c699b48aSwangkaifan    base + 0x00217F: Enable bits for sources 992-1023 on context 2
45c699b48aSwangkaifan    ...
46c699b48aSwangkaifan    base + 0x1F1F80: Enable bits for sources 0-31 on context 15871
47c699b48aSwangkaifan    base + 0x1F1F84: Enable bits for sources 32-63 on context 15871
48c699b48aSwangkaifan    base + 0x1F1FFF: Enable bits for sources 992-1023 on context 15871
49c699b48aSwangkaifan    ...
50c699b48aSwangkaifan    base + 0x1FFFFC: Reserved
51c699b48aSwangkaifan    base + 0x200000: Priority threshold for context 0
52c699b48aSwangkaifan    base + 0x200004: Claim/complete for context 0
53c699b48aSwangkaifan    base + 0x200008: Reserved
54c699b48aSwangkaifan    ...
55c699b48aSwangkaifan    base + 0x200FFC: Reserved
56c699b48aSwangkaifan    base + 0x201000: Priority threshold for context 1
57c699b48aSwangkaifan    base + 0x201004: Claim/complete for context 1
58c699b48aSwangkaifan    ...
59c699b48aSwangkaifan    base + 0x3FFE000: Priority threshold for context 15871
60c699b48aSwangkaifan    base + 0x3FFE004: Claim/complete for context 15871
61c699b48aSwangkaifan    base + 0x3FFE008: Reserved
62c699b48aSwangkaifan    ...
63c699b48aSwangkaifan    base + 0x3FFFFFC: Reserved  */
64c699b48aSwangkaifan
65c699b48aSwangkaifanobject PLICConsts
66c699b48aSwangkaifan{
67c699b48aSwangkaifan  def maxDevices = 1023
68c699b48aSwangkaifan  def maxHarts = 15872
69c699b48aSwangkaifan  def priorityBase = 0x0
70c699b48aSwangkaifan  def pendingBase = 0x1000
71c699b48aSwangkaifan  def enableBase = 0x2000
72c699b48aSwangkaifan  def hartBase = 0x200000
73c699b48aSwangkaifan
74c699b48aSwangkaifan  def claimOffset = 4
75c699b48aSwangkaifan  def priorityBytes = 4
76c699b48aSwangkaifan
77c699b48aSwangkaifan  def enableOffset(i: Int) = i * ((maxDevices+7)/8)
78c699b48aSwangkaifan  def hartOffset(i: Int) = i * 0x1000
79c699b48aSwangkaifan  def enableBase(i: Int):Int = enableOffset(i) + enableBase
80c699b48aSwangkaifan  def hartBase(i: Int):Int = hartOffset(i) + hartBase
81c699b48aSwangkaifan
82c699b48aSwangkaifan  def size(maxHarts: Int): Int = {
83c699b48aSwangkaifan    require(maxHarts > 0 && maxHarts <= maxHarts, s"Must be: maxHarts=$maxHarts > 0 && maxHarts <= PLICConsts.maxHarts=${PLICConsts.maxHarts}")
84c699b48aSwangkaifan    1 << log2Ceil(hartBase(maxHarts))
85c699b48aSwangkaifan  }
86c699b48aSwangkaifan
87c699b48aSwangkaifan  require(hartBase >= enableBase(maxHarts))
88c699b48aSwangkaifan}
89c699b48aSwangkaifan
902225d46eSJiawei Linclass PlicIO(val numCores: Int, val numExtIntrs: Int) extends Bundle{
912225d46eSJiawei Lin  val intrVec = Input(UInt(numExtIntrs.W))
922225d46eSJiawei Lin  val meip = Output(Vec(numCores, Bool()))
93c699b48aSwangkaifan}
94c699b48aSwangkaifan
95c699b48aSwangkaifanclass AXI4Plic
96c699b48aSwangkaifan(
974a26299eSwangkaifan  address: Seq[AddressSet],
982225d46eSJiawei Lin  numCores: Int,
992225d46eSJiawei Lin  numExtIntrs: Int,
1004a26299eSwangkaifan  sim: Boolean = false
101c699b48aSwangkaifan)(implicit p: Parameters)
1022225d46eSJiawei Lin  extends AXI4SlaveModule(address, executable = false, _extra = new PlicIO(numCores, numExtIntrs))
103c699b48aSwangkaifan{
104c699b48aSwangkaifan  override lazy val module = new AXI4SlaveModuleImp[PlicIO](this) {
1052225d46eSJiawei Lin    require(numCores <= PLICConsts.maxDevices)
1062225d46eSJiawei Lin    require(numExtIntrs <= PLICConsts.maxHarts)
107c699b48aSwangkaifan    val addressSpaceSize = 0x4000000
108c699b48aSwangkaifan    val addressBits = log2Up(addressSpaceSize)
109c699b48aSwangkaifan
110c699b48aSwangkaifan    def getOffset(addr: UInt) = addr(addressBits - 1, 0)
111c699b48aSwangkaifan
1122225d46eSJiawei Lin    val priority = List.fill(numExtIntrs)(Reg(UInt(32.W)))
113c699b48aSwangkaifan    val priorityMap = priority.zipWithIndex.map { case (r, intr) => RegMap((intr + 1) * 4, r) }.toMap
114c699b48aSwangkaifan
1152225d46eSJiawei Lin    val nrIntrWord = (numExtIntrs + 31) / 32 // roundup
116c699b48aSwangkaifan    // pending bits are updated in the unit of bit by PLIC,
117c699b48aSwangkaifan    // so define it as vectors of bits, instead of UInt(32.W)
118c699b48aSwangkaifan    val pending = List.fill(nrIntrWord)(RegInit(0.U.asTypeOf(Vec(32, Bool()))))
119c699b48aSwangkaifan    val pendingMap = pending.zipWithIndex.map { case (r, intrWord) =>
120c699b48aSwangkaifan      RegMap(0x1000 + intrWord * 4, Cat(r.reverse), RegMap.Unwritable)
121c699b48aSwangkaifan    }.toMap
122c699b48aSwangkaifan
1232225d46eSJiawei Lin    val enable = List.fill(numCores)(List.fill(nrIntrWord)(RegInit(0.U(32.W))))
124c699b48aSwangkaifan    val enableMap = enable.zipWithIndex.map { case (l, hart) =>
125c699b48aSwangkaifan      l.zipWithIndex.map { case (r, intrWord) => RegMap(0x2000 + hart * 0x80 + intrWord * 4, r) }
126c699b48aSwangkaifan    }.reduce(_ ++ _).toMap
127c699b48aSwangkaifan
1282225d46eSJiawei Lin    val threshold = List.fill(numCores)(Reg(UInt(32.W)))
129c699b48aSwangkaifan    val thresholdMap = threshold.zipWithIndex.map {
130c699b48aSwangkaifan      case (r, hart) => RegMap(0x200000 + hart * 0x1000, r)
131c699b48aSwangkaifan    }.toMap
132c699b48aSwangkaifan
1332225d46eSJiawei Lin    val inHandle = RegInit(0.U.asTypeOf(Vec(numExtIntrs + 1, Bool())))
134c699b48aSwangkaifan
135c699b48aSwangkaifan    def completionFn(wdata: UInt) = {
136c699b48aSwangkaifan      inHandle(wdata(31, 0)) := false.B
137c699b48aSwangkaifan      0.U
138c699b48aSwangkaifan    }
139c699b48aSwangkaifan
1402225d46eSJiawei Lin    val claimCompletion = List.fill(numCores)(Reg(UInt(32.W)))
141c699b48aSwangkaifan    val claimCompletionMap = claimCompletion.zipWithIndex.map {
142c699b48aSwangkaifan      case (r, hart) => {
143c699b48aSwangkaifan        val addr = 0x200004 + hart * 0x1000
144c699b48aSwangkaifan        when(in.r.fire() && (getOffset(raddr) === addr.U)) {
145c699b48aSwangkaifan          inHandle(r) := true.B
146c699b48aSwangkaifan        }
147c699b48aSwangkaifan        RegMap(addr, r, completionFn)
148c699b48aSwangkaifan      }
149c699b48aSwangkaifan    }.toMap
150c699b48aSwangkaifan
1512225d46eSJiawei Lin    val intrVecReg = Wire(UInt(numExtIntrs.W))
152c0bc1ee4SYinan Xu    intrVecReg := RegNext(RegNext(RegNext(io.extra.get.intrVec)))
153c0bc1ee4SYinan Xu    intrVecReg.asBools.zipWithIndex.map { case (intr, i) => {
154c699b48aSwangkaifan      val id = i + 1
155c699b48aSwangkaifan      when(intr) {
156c699b48aSwangkaifan        pending(id / 32)(id % 32) := true.B
157c699b48aSwangkaifan      }
158c699b48aSwangkaifan      when(inHandle(id)) {
159c699b48aSwangkaifan        pending(id / 32)(id % 32) := false.B
160c699b48aSwangkaifan      }
161c699b48aSwangkaifan    }
162c699b48aSwangkaifan    }
163c699b48aSwangkaifan
164c699b48aSwangkaifan    val pendingVec = Cat(pending.map(x => Cat(x.reverse)))
165c699b48aSwangkaifan    claimCompletion.zipWithIndex.map { case (r, hart) => {
166c699b48aSwangkaifan      val takenVec = pendingVec & Cat(enable(hart))
167c699b48aSwangkaifan      r := Mux(takenVec === 0.U, 0.U, PriorityEncoder(takenVec))
168c699b48aSwangkaifan    }
169c699b48aSwangkaifan    }
170c699b48aSwangkaifan
171c699b48aSwangkaifan    val mapping = priorityMap ++ pendingMap ++ enableMap ++ thresholdMap ++ claimCompletionMap
172c699b48aSwangkaifan
173c699b48aSwangkaifan    val rdata = Wire(UInt(32.W))
174c699b48aSwangkaifan    RegMap.generate(mapping, getOffset(raddr), rdata,
175c699b48aSwangkaifan      getOffset(waddr), in.w.fire(), in.w.bits.data, MaskExpand(in.w.bits.strb >> waddr(2, 0)))
176c699b48aSwangkaifan    // narrow read
177c699b48aSwangkaifan    in.r.bits.data := Fill(2, rdata)
178c699b48aSwangkaifan
179c699b48aSwangkaifan    io.extra.get.meip.zipWithIndex.map { case (ip, hart) => ip := claimCompletion(hart) =/= 0.U }
180c699b48aSwangkaifan  }
181c699b48aSwangkaifan}
182