summaryrefslogtreecommitdiff
path: root/src/test/scala/chiselTests/VecLiteralSpec.scala
diff options
context:
space:
mode:
Diffstat (limited to 'src/test/scala/chiselTests/VecLiteralSpec.scala')
-rw-r--r--src/test/scala/chiselTests/VecLiteralSpec.scala529
1 files changed, 0 insertions, 529 deletions
diff --git a/src/test/scala/chiselTests/VecLiteralSpec.scala b/src/test/scala/chiselTests/VecLiteralSpec.scala
deleted file mode 100644
index dcc96b17..00000000
--- a/src/test/scala/chiselTests/VecLiteralSpec.scala
+++ /dev/null
@@ -1,529 +0,0 @@
-// SPDX-License-Identifier: Apache-2.0
-
-package chiselTests
-
-import chisel3._
-import chisel3.experimental.BundleLiterals.AddBundleLiteralConstructor
-import chisel3.experimental.VecLiterals._
-import chisel3.experimental.{FixedPoint, VecLiteralException}
-import chisel3.stage.ChiselStage
-import chisel3.testers.BasicTester
-import chisel3.util.Counter
-import scala.language.reflectiveCalls
-
-class VecLiteralSpec extends ChiselFreeSpec with Utils {
- object MyEnum extends ChiselEnum {
- val sA, sB, sC = Value
- }
- object MyEnumB extends ChiselEnum {
- val sA, sB = Value
- }
-
- "Vec literals should work with chisel Enums" in {
- val enumVec = Vec(3, MyEnum()).Lit(0 -> MyEnum.sA, 1 -> MyEnum.sB, 2 -> MyEnum.sC)
- enumVec(0).toString should include(MyEnum.sA.toString)
- enumVec(1).toString should include(MyEnum.sB.toString)
- enumVec(2).toString should include(MyEnum.sC.toString)
- }
-
- "improperly constructed vec literals should be detected" - {
- "indices in vec literal muse be greater than zero and less than length" in {
- val e = intercept[VecLiteralException] {
- Vec(2, UInt(4.W)).Lit(0 -> 1.U, 1 -> 2.U, 2 -> 3.U, 3 -> 4.U, -2 -> 7.U)
- }
- e.getMessage should include(
- "VecLiteral: The following indices (2,3,-2) are less than zero or greater or equal to than Vec length"
- )
- }
-
- "indices in vec literals must not be repeated" in {
- val e = intercept[VecLiteralException] {
- Vec(2, UInt(4.W)).Lit(0 -> 1.U, 1 -> 2.U, 2 -> 3.U, 2 -> 3.U, 2 -> 3.U, 3 -> 4.U)
- }
- e.getMessage should include("VecLiteral: has duplicated indices 2(3 times)")
- }
- "lits must fit in vec element width" in {
- val e = intercept[VecLiteralException] {
- Vec(2, SInt(4.W)).Lit(0 -> 0xab.S, 1 -> 0xbc.S)
- }
- e.getMessage should include(
- "VecLiteral: Vec[SInt<4>] has the following incorrectly typed or sized initializers: " +
- "0 -> SInt<9>(171),1 -> SInt<9>(188)"
- )
- }
-
- "all lits must be the same type but width can be equal or smaller than the Vec's element width" in {
- val v = Vec(2, SInt(4.W)).Lit(0 -> 1.S, 1 -> -2.S)
- v(0).toString should include(1.S(4.W).toString)
- v(1).toString should include((-2).S(4.W).toString)
- v.toString should include("SInt<4>[2](0=SLit(1,<4>), 1=SLit(-2,<4>)")
- }
-
- "all lits must be the same type but width cannot be greater than Vec's element width" in {
- val e = intercept[VecLiteralException] {
- val v = Vec(2, SInt(4.W)).Lit(0 -> 11.S, 1 -> -0xffff.S)
- }
- e.getMessage should include(
- "VecLiteral: Vec[SInt<4>] has the following incorrectly typed or sized initializers: 0 -> SInt<5>(11),1 -> SInt<17>(-65535)"
- )
- }
- }
-
- //NOTE: I had problems where this would not work if this class declaration was inside test scope
- class HasVecInit extends Module {
- val initValue = Vec(4, UInt(8.W)).Lit(0 -> 0xab.U(8.W), 1 -> 0xcd.U(8.W), 2 -> 0xef.U(8.W), 3 -> 0xff.U(8.W))
- val y = RegInit(initValue)
- }
-
- "Vec literals should work when used to initialize a reg of vec" in {
- val firrtl = (new ChiselStage).emitFirrtl(new HasVecInit, args = Array("--full-stacktrace"))
- firrtl should include("""_y_WIRE[0] <= UInt<8>("hab")""")
- firrtl should include("""_y_WIRE[1] <= UInt<8>("hcd")""")
- firrtl should include("""_y_WIRE[2] <= UInt<8>("hef")""")
- firrtl should include("""_y_WIRE[3] <= UInt<8>("hff")""")
- firrtl should include(""" reset => (reset, _y_WIRE)""".stripMargin)
- }
-
- //NOTE: I had problems where this would not work if this class declaration was inside test scope
- class HasPartialVecInit extends Module {
- val initValue = Vec(4, UInt(8.W)).Lit(0 -> 0xab.U(8.W), 2 -> 0xef.U(8.W), 3 -> 0xff.U(8.W))
- val y = RegInit(initValue)
- }
-
- "Vec literals should work when used to partially initialize a reg of vec" in {
- val firrtl = (new ChiselStage).emitFirrtl(new HasPartialVecInit, args = Array("--full-stacktrace"))
- firrtl should include("""_y_WIRE[0] <= UInt<8>("hab")""")
- firrtl should include("""_y_WIRE[1] is invalid""")
- firrtl should include("""_y_WIRE[2] <= UInt<8>("hef")""")
- firrtl should include("""_y_WIRE[3] <= UInt<8>("hff")""")
- firrtl should include(""" reset => (reset, _y_WIRE)""".stripMargin)
- }
-
- class ResetRegWithPartialVecLiteral extends Module {
- val in = IO(Input(Vec(4, UInt(8.W))))
- val out = IO(Output(Vec(4, UInt(8.W))))
- val initValue = Vec(4, UInt(8.W)).Lit(0 -> 0xab.U(8.W), 2 -> 0xef.U(8.W), 3 -> 0xff.U(8.W))
- val y = RegInit(initValue)
- when(in(1) > 0.U) {
- y(1) := in(1)
- }
- when(in(2) > 0.U) {
- y(2) := in(2)
- }
- out := y
- }
-
- "Vec literals should only init specified fields when used to partially initialize a reg of vec" in {
- println(ChiselStage.emitFirrtl(new ResetRegWithPartialVecLiteral))
- assertTesterPasses(new BasicTester {
- val m = Module(new ResetRegWithPartialVecLiteral)
- val (counter, wrapped) = Counter(true.B, 8)
- m.in := DontCare
- when(counter < 2.U) {
- m.in(1) := 0xff.U
- m.in(2) := 0xff.U
- }.elsewhen(counter === 2.U) {
- chisel3.assert(m.out(1) === 0xff.U)
- chisel3.assert(m.out(2) === 0xff.U)
- }.elsewhen(counter === 3.U) {
- m.in(1) := 0.U
- m.in(2) := 0.U
- m.reset := true.B
- }.elsewhen(counter > 2.U) {
- // m.out(1) should not be reset, m.out(2) should be reset
- chisel3.assert(m.out(1) === 0xff.U)
- chisel3.assert(m.out(2) === 0xef.U)
- }
- when(wrapped) {
- stop()
- }
- })
- }
-
- "lowest of vec literal contains least significant bits and " in {
- val y = Vec(4, UInt(8.W)).Lit(0 -> 0xab.U(8.W), 1 -> 0xcd.U(8.W), 2 -> 0xef.U(8.W), 3 -> 0xff.U(8.W))
- y.litValue should be(BigInt("FFEFCDAB", 16))
- }
-
- "the order lits are specified does not matter" in {
- val y = Vec(4, UInt(8.W)).Lit(3 -> 0xff.U(8.W), 2 -> 0xef.U(8.W), 1 -> 0xcd.U(8.W), 0 -> 0xab.U(8.W))
- y.litValue should be(BigInt("FFEFCDAB", 16))
- }
-
- "regardless of the literals widths, packing should be done based on the width of the Vec's gen" in {
- val z = Vec(4, UInt(8.W)).Lit(0 -> 0x2.U, 1 -> 0x2.U, 2 -> 0x2.U, 3 -> 0x3.U)
- z.litValue should be(BigInt("03020202", 16))
- }
-
- "packing sparse vec lits should not pack, litOption returns None" in {
- // missing sub-listeral for index 2
- val z = Vec(4, UInt(8.W)).Lit(0 -> 0x2.U, 1 -> 0x2.U, 3 -> 0x3.U)
-
- z.litOption should be(None)
- }
-
- "registers can be initialized with a Vec literal" in {
- assertTesterPasses(new BasicTester {
- val y = RegInit(Vec(4, UInt(8.W)).Lit(0 -> 0xab.U(8.W), 1 -> 0xcd.U(8.W), 2 -> 0xef.U(8.W), 3 -> 0xff.U(8.W)))
- chisel3.assert(y.asUInt === BigInt("FFEFCDAB", 16).U)
- stop()
- })
- }
-
- "how does asUInt work" in {
- assertTesterPasses(new BasicTester {
- val vec1 = Vec(4, UInt(16.W)).Lit(0 -> 0xdd.U, 1 -> 0xcc.U, 2 -> 0xbb.U, 3 -> 0xaa.U)
-
- val vec2 = VecInit(Seq(0xdd.U, 0xcc.U, 0xbb.U, 0xaa.U))
- printf("vec1 %x\n", vec1.asUInt())
- printf("vec2 %x\n", vec2.asUInt())
- stop()
- })
- }
-
- "Vec literals uint conversion" in {
- class M1 extends Module {
- val out1 = IO(Output(UInt(64.W)))
- val out2 = IO(Output(UInt(64.W)))
-
- val v1 = Vec(4, UInt(16.W)).Lit(0 -> 0xdd.U, 1 -> 0xcc.U, 2 -> 0xbb.U, 3 -> 0xaa.U)
- out1 := v1.asUInt
-
- val v2 = VecInit(0xdd.U(16.W), 0xcc.U, 0xbb.U, 0xaa.U)
- out2 := v2.asUInt
- }
-
- assertTesterPasses(new BasicTester {
- val m = Module(new M1)
- chisel3.assert(m.out1 === m.out2)
- stop()
- })
- }
-
- "VecLits should work properly with .asUInt" in {
- val outsideVecLit = Vec(4, UInt(16.W)).Lit(0 -> 0xdd.U, 1 -> 0xcc.U, 2 -> 0xbb.U, 3 -> 0xaa.U)
-
- assertTesterPasses {
- new BasicTester {
- chisel3.assert(outsideVecLit(0) === 0xdd.U, "v(0)")
- stop()
- }
- }
- }
-
- "bundle literals should work in RTL" in {
- val outsideVecLit = Vec(4, UInt(16.W)).Lit(0 -> 0xdd.U, 1 -> 0xcc.U, 2 -> 0xbb.U, 3 -> 0xaa.U)
-
- assertTesterPasses {
- new BasicTester {
- chisel3.assert(outsideVecLit(0) === 0xdd.U, "v(0)")
- chisel3.assert(outsideVecLit(1) === 0xcc.U)
- chisel3.assert(outsideVecLit(2) === 0xbb.U)
- chisel3.assert(outsideVecLit(3) === 0xaa.U)
-
- chisel3.assert(outsideVecLit.litValue.U === outsideVecLit.asUInt())
-
- val insideVecLit = Vec(4, UInt(16.W)).Lit(0 -> 0xdd.U, 1 -> 0xcc.U, 2 -> 0xbb.U, 3 -> 0xaa.U)
- chisel3.assert(insideVecLit(0) === 0xdd.U)
- chisel3.assert(insideVecLit(1) === 0xcc.U)
- chisel3.assert(insideVecLit(2) === 0xbb.U)
- chisel3.assert(insideVecLit(3) === 0xaa.U)
-
- chisel3.assert(insideVecLit(0) === outsideVecLit(0))
- chisel3.assert(insideVecLit(1) === outsideVecLit(1))
- chisel3.assert(insideVecLit(2) === outsideVecLit(2))
- chisel3.assert(insideVecLit(3) === outsideVecLit(3))
-
- val vecWire1 = Wire(Vec(4, UInt(16.W)))
- vecWire1 := outsideVecLit
-
- chisel3.assert(vecWire1(0) === 0xdd.U)
- chisel3.assert(vecWire1(1) === 0xcc.U)
- chisel3.assert(vecWire1(2) === 0xbb.U)
- chisel3.assert(vecWire1(3) === 0xaa.U)
-
- val vecWire2 = Wire(Vec(4, UInt(16.W)))
- vecWire2 := insideVecLit
-
- chisel3.assert(vecWire2(0) === 0xdd.U)
- chisel3.assert(vecWire2(1) === 0xcc.U)
- chisel3.assert(vecWire2(2) === 0xbb.U)
- chisel3.assert(vecWire2(3) === 0xaa.U)
-
- stop()
- }
- }
- }
-
- "partial vec literals should work in RTL" in {
- assertTesterPasses {
- new BasicTester {
- val vecLit = Vec(4, UInt(8.W)).Lit(0 -> 42.U, 2 -> 5.U)
- chisel3.assert(vecLit(0) === 42.U)
- chisel3.assert(vecLit(2) === 5.U)
-
- val vecWire = Wire(Vec(4, UInt(8.W)))
- vecWire := vecLit
-
- chisel3.assert(vecWire(0) === 42.U)
- chisel3.assert(vecWire(2) === 5.U)
-
- stop()
- }
- }
- }
-
- "nested vec literals should be constructable" in {
- val outerVec = Vec(2, Vec(3, UInt(4.W))).Lit(
- 0 -> Vec(3, UInt(4.W)).Lit(0 -> 1.U, 1 -> 2.U, 2 -> 3.U),
- 1 -> Vec(3, UInt(4.W)).Lit(0 -> 4.U, 1 -> 5.U, 2 -> 6.U)
- )
-
- outerVec.litValue should be(BigInt("654321", 16))
- outerVec(0).litValue should be(BigInt("321", 16))
- outerVec(1).litValue should be(BigInt("654", 16))
- outerVec(0)(0).litValue should be(BigInt(1))
- outerVec(0)(1).litValue should be(BigInt(2))
- outerVec(0)(2).litValue should be(BigInt(3))
- outerVec(1)(0).litValue should be(BigInt(4))
- outerVec(1)(1).litValue should be(BigInt(5))
- outerVec(1)(2).litValue should be(BigInt(6))
- }
-
- "contained vecs should work" in {
- assertTesterPasses {
- new BasicTester {
- val outerVec = Vec(2, Vec(3, UInt(4.W))).Lit(
- 0 -> Vec(3, UInt(4.W)).Lit(0 -> 1.U, 1 -> 2.U, 2 -> 3.U),
- 1 -> Vec(3, UInt(4.W)).Lit(0 -> 4.U, 1 -> 5.U, 2 -> 6.U)
- )
-
- chisel3.assert(outerVec(0)(0) === 1.U)
- chisel3.assert(outerVec(0)(1) === 2.U)
- chisel3.assert(outerVec(0)(2) === 3.U)
- chisel3.assert(outerVec(1)(0) === 4.U)
- chisel3.assert(outerVec(1)(1) === 5.U)
- chisel3.assert(outerVec(1)(2) === 6.U)
-
- val v0 = outerVec(0)
- val v1 = outerVec(1)
- chisel3.assert(v0(0) === 1.U)
- chisel3.assert(v0(1) === 2.U)
- chisel3.assert(v0(2) === 3.U)
- chisel3.assert(v1(0) === 4.U)
- chisel3.assert(v1(1) === 5.U)
- chisel3.assert(v1(2) === 6.U)
-
- stop()
- }
- }
- }
-
- //TODO: decide what behavior here should be
- "This doesn't work should it" ignore {
- assertTesterPasses {
- new BasicTester {
- def vecFactory = Vec(2, FixedPoint(8.W, 4.BP))
-
- val vecWire1 = Wire(Output(vecFactory))
- val vecLit1 = vecFactory.Lit(0 -> (1.5).F(8.W, 4.BP))
- val vecLit2 = vecFactory.Lit(1 -> (3.25).F(8.W, 4.BP))
-
- vecWire1 := vecLit1
- vecWire1 := vecLit2
- printf("vw1(0) %x vw1(1) %x\n", vecWire1(0).asUInt(), vecWire1(1).asUInt())
- chisel3.assert(vecWire1(0) === (1.5).F(8.W, 4.BP))
- chisel3.assert(vecWire1(1) === (3.25).F(8.W, 4.BP))
- stop()
- }
- }
- }
-
- "partially initialized Vec literals should assign" in {
- assertTesterPasses {
- new BasicTester {
- def vecFactory = Vec(2, FixedPoint(8.W, 4.BP))
-
- val vecWire1 = Wire(Output(vecFactory))
- val vecWire2 = Wire(Output(vecFactory))
- val vecLit1 = vecFactory.Lit(0 -> (1.5).F(8.W, 4.BP))
- val vecLit2 = vecFactory.Lit(1 -> (3.25).F(8.W, 4.BP))
-
- vecWire1 := vecLit1
- vecWire2 := vecLit2
- vecWire1(1) := (0.5).F(8.W, 4.BP)
- printf("vw1(0) %x vw1(1) %x\n", vecWire1(0).asUInt(), vecWire1(1).asUInt())
- chisel3.assert(vecWire1(0) === (1.5).F(8.W, 4.BP))
- chisel3.assert(vecWire1(1) === (0.5).F(8.W, 4.BP)) // Last connect won
- chisel3.assert(vecWire2(1) === (3.25).F(8.W, 4.BP))
- stop()
- }
- }
- }
-
- "Vec literals should work as register reset values" in {
- assertTesterPasses {
- new BasicTester {
- val r = RegInit(Vec(3, UInt(11.W)).Lit(0 -> 0xa.U, 1 -> 0xb.U, 2 -> 0xc.U))
- r := (r.asUInt + 1.U).asTypeOf(Vec(3, UInt(11.W))) // prevent constprop
-
- // check reset values on first cycle out of reset
- chisel3.assert(r(0) === 0xa.U)
- chisel3.assert(r(1) === 0xb.U)
- chisel3.assert(r(2) === 0xc.U)
- stop()
- }
- }
- }
-
- "partially initialized Vec literals should work as register reset values" in {
- assertTesterPasses {
- new BasicTester {
- val r = RegInit(Vec(3, UInt(11.W)).Lit(0 -> 0xa.U, 2 -> 0xc.U))
- r := (r.asUInt + 1.U).asTypeOf(Vec(3, UInt(11.W))) // prevent constprop
- // check reset values on first cycle out of reset
- chisel3.assert(r(0) === 0xa.U)
- chisel3.assert(r(2) === 0xc.U)
- stop()
- }
- }
- }
-
- "Fields extracted from Vec Literals should work as register reset values" in {
- assertTesterPasses {
- new BasicTester {
- val r = RegInit(Vec(3, UInt(11.W)).Lit(0 -> 0xa.U, 2 -> 0xc.U).apply(0))
- r := r + 1.U // prevent const prop
- chisel3.assert(r === 0xa.U) // coming out of reset
- stop()
- }
- }
- }
-
- "DontCare fields extracted from Vec Literals should work as register reset values" in {
- assertTesterPasses {
- new BasicTester {
- val r = RegInit(Vec(3, Bool()).Lit(0 -> true.B).apply(2))
- r := reset.asBool
- printf(p"r = $r\n") // Can't assert because reset value is DontCare
- stop()
- }
- }
- }
-
- "DontCare fields extracted from Vec Literals should work in other Expressions" in {
- assertTesterPasses {
- new BasicTester {
- val x = Vec(3, Bool()).Lit(0 -> true.B).apply(2) || true.B
- chisel3.assert(x === true.B)
- stop()
- }
- }
- }
-
- "vec literals with non-literal values should fail" in {
- val exc = intercept[VecLiteralException] {
- extractCause[VecLiteralException] {
- ChiselStage.elaborate {
- new RawModule {
- (Vec(3, UInt(11.W)).Lit(0 -> UInt()))
- }
- }
- }
- }
- exc.getMessage should include("field 0 specified with non-literal value UInt")
- }
-
- "vec literals are instantiated on connect and are not bulk connected" in {
- class VecExample5 extends RawModule {
- val out = IO(Output(Vec(2, UInt(4.W))))
- val bundle = Vec(2, UInt(4.W)).Lit(
- 0 -> 0xa.U,
- 1 -> 0xb.U
- )
- out := bundle
- }
-
- val firrtl = (new chisel3.stage.ChiselStage).emitFirrtl(new VecExample5, args = Array("--full-stacktrace"))
- firrtl should include("""out[0] <= UInt<4>("ha")""")
- firrtl should include("""out[1] <= UInt<4>("hb")""")
- }
-
- class SubBundle extends Bundle {
- val foo = UInt(8.W)
- val bar = UInt(4.W)
- }
-
- class VecExample extends RawModule {
- val out = IO(Output(Vec(2, new SubBundle)))
- val bundle = Vec(2, new SubBundle).Lit(
- 0 -> (new SubBundle).Lit(_.foo -> 42.U, _.bar -> 22.U),
- 1 -> (new SubBundle).Lit(_.foo -> 7.U, _.bar -> 3.U)
- )
- out := bundle
- }
-
- "vec literals can contain bundles and should not be bulk connected" in {
- val chirrtl = (new chisel3.stage.ChiselStage).emitChirrtl(new VecExample, args = Array("--full-stacktrace"))
- chirrtl should include("""out[0].bar <= UInt<5>("h16")""")
- chirrtl should include("""out[0].foo <= UInt<6>("h2a")""")
- chirrtl should include("""out[1].bar <= UInt<2>("h3")""")
- chirrtl should include("""out[1].foo <= UInt<3>("h7")""")
- }
-
- "vec literals can have bundle children" in {
- val vec = Vec(2, new SubBundle).Lit(
- 0 -> (new SubBundle).Lit(_.foo -> 0xab.U, _.bar -> 0xc.U),
- 1 -> (new SubBundle).Lit(_.foo -> 0xde.U, _.bar -> 0xf.U)
- )
- vec.litValue.toString(16) should be("defabc")
- }
-
- "vec literals can have bundle children assembled incrementally" in {
- val bundle1 = (new SubBundle).Lit(_.foo -> 0xab.U, _.bar -> 0xc.U)
- val bundle2 = (new SubBundle).Lit(_.foo -> 0xde.U, _.bar -> 0xf.U)
-
- bundle1.litValue.toString(16) should be("abc")
- bundle2.litValue.toString(16) should be("def")
-
- val vec = Vec(2, new SubBundle).Lit(0 -> bundle1, 1 -> bundle2)
-
- vec.litValue.toString(16) should be("defabc")
- }
-
- "bundles can contain vec lits" in {
- val vec1 = Vec(3, UInt(4.W)).Lit(0 -> 0xa.U, 1 -> 0xb.U, 2 -> 0xc.U)
- val vec2 = Vec(2, UInt(4.W)).Lit(0 -> 0xd.U, 1 -> 0xe.U)
- val bundle = (new Bundle {
- val foo = Vec(3, UInt(4.W))
- val bar = Vec(2, UInt(4.W))
- }).Lit(_.foo -> vec1, _.bar -> vec2)
- bundle.litValue.toString(16) should be("cbaed")
- }
-
- "bundles can contain vec lits in-line" in {
- val bundle = (new Bundle {
- val foo = Vec(3, UInt(4.W))
- val bar = Vec(2, UInt(4.W))
- }).Lit(
- _.foo -> Vec(3, UInt(4.W)).Lit(0 -> 0xa.U, 1 -> 0xb.U, 2 -> 0xc.U),
- _.bar -> Vec(2, UInt(4.W)).Lit(0 -> 0xd.U, 1 -> 0xe.U)
- )
- bundle.litValue.toString(16) should be("cbaed")
- }
-
- "Vec.Lit is a trivial Vec literal factory" in {
- val vec = Vec.Lit(0xa.U, 0xb.U)
- vec(0).litValue should be(0xa)
- vec(1).litValue should be(0xb)
- }
-
- "Vec.Lit bases it's element width on the widest literal supplied" in {
- val vec = Vec.Lit(0xa.U, 0xbbbb.U)
- vec(0).litValue should be(0xa)
- vec(1).litValue should be(0xbbbb)
- vec.length should be(2)
- vec.getWidth should be(16 * 2)
- vec.litValue should be(BigInt("bbbb000a", 16))
- }
-}