Add option to disable VCore.

Change-Id: Ie139a62fb0ac755dc5fbbec61b8b7aa68c17425e
diff --git a/hdl/chisel/src/kelvin/Core.scala b/hdl/chisel/src/kelvin/Core.scala
index 100078c..72b1c03 100644
--- a/hdl/chisel/src/kelvin/Core.scala
+++ b/hdl/chisel/src/kelvin/Core.scala
@@ -34,7 +34,9 @@
 
     val ibus = new IBusIO(p)
     val dbus = new DBusIO(p)
-    val axi0 = new AxiMasterIO(p.axi2AddrBits, p.axi2DataBits, p.axi2IdBits)
+    val axi0 = if(p.enableVector) {
+      Some(new AxiMasterIO(p.axi2AddrBits, p.axi2DataBits, p.axi2IdBits))
+    } else { None }
     val axi1 = new AxiMasterIO(p.axi2AddrBits, p.axi2DataBits, p.axi2IdBits)
 
     val iflush = new IFlushIO(p)
@@ -45,8 +47,7 @@
   })
 
   val score = SCore(p)
-  val vcore = VCore(p)
-  val dbusmux = DBusMux(p)
+  val vcore = if (p.enableVector) { Some(VCore(p)) } else { None }
 
   // ---------------------------------------------------------------------------
   // Scalar Core outputs.
@@ -61,17 +62,22 @@
 
   // ---------------------------------------------------------------------------
   // Vector core.
-  score.io.vcore <> vcore.io.score
+  if (p.enableVector) {
+    score.io.vcore.get <> vcore.get.io.score
+  }
 
   // ---------------------------------------------------------------------------
   // Local Data Bus Port
-  dbusmux.io.vldst := score.io.vldst
-  dbusmux.io.vlast := vcore.io.last
-
-  dbusmux.io.vcore <> vcore.io.dbus
-  dbusmux.io.score <> score.io.dbus
-
-  io.dbus <> dbusmux.io.dbus
+  if (p.enableVector) {
+    val dbusmux = DBusMux(p)
+    dbusmux.io.vldst := score.io.vldst.get
+    dbusmux.io.vlast := vcore.get.io.last
+    dbusmux.io.vcore <> vcore.get.io.dbus
+    dbusmux.io.score <> score.io.dbus
+    io.dbus <> dbusmux.io.dbus
+  } else {
+    io.dbus <> score.io.dbus
+  }
 
   // ---------------------------------------------------------------------------
   // Scalar DBus to AXI.
@@ -80,8 +86,10 @@
 
   // ---------------------------------------------------------------------------
   // AXI ports.
-  io.axi0.read  <> vcore.io.ld
-  io.axi0.write <> vcore.io.st
+  if (p.enableVector) {
+    io.axi0.get.read  <> vcore.get.io.ld
+    io.axi0.get.write <> vcore.get.io.st
+  }
 
   io.axi1 <> dbus2axi.io.axi
 }
diff --git a/hdl/chisel/src/kelvin/Parameters.scala b/hdl/chisel/src/kelvin/Parameters.scala
index 3546a74..a93f69e 100644
--- a/hdl/chisel/src/kelvin/Parameters.scala
+++ b/hdl/chisel/src/kelvin/Parameters.scala
@@ -56,6 +56,8 @@
 
   val vectorCountBits = log2Ceil(vectorBits / 8) + 1 + 2  // +2 stripmine
 
+  // Enable Vector
+  val enableVector = true
   val vectorAluCount = 2
   val vectorReadPorts = (vectorAluCount * 3) + 1
   val vectorWritePorts = 6
diff --git a/hdl/chisel/src/kelvin/scalar/Csr.scala b/hdl/chisel/src/kelvin/scalar/Csr.scala
index 18202d3..621de70 100644
--- a/hdl/chisel/src/kelvin/scalar/Csr.scala
+++ b/hdl/chisel/src/kelvin/scalar/Csr.scala
@@ -47,8 +47,12 @@
   val rfwriteCount = UInt(3.W)
   val storeCount = UInt(2.W)
   val branchCount = UInt(1.W)
-  val vrfwriteCount = UInt(3.W)
-  val vstoreCount = UInt(2.W)
+  val vrfwriteCount = if (p.enableVector) {
+    Some(UInt(3.W))
+  } else { None }
+  val vstoreCount = if (p.enableVector) {
+    Some(UInt(2.W))
+  } else { None }
 }
 
 class CsrBruIO(p: Parameters) extends Bundle {
@@ -92,7 +96,9 @@
     val bru = Flipped(new CsrBruIO(p))
 
     // Vector core.
-    val vcore = Input(new Bundle { val undef = Bool() })
+    val vcore = (if (p.enableVector) {
+      Some(Input(new Bundle { val undef = Bool() }))
+    } else { None })
 
     val counters = Input(new CsrCounters(p))
 
@@ -187,13 +193,15 @@
   val kisaEn      = req.bits.index === 0xFC0.U
 
   // Pipeline Control.
-  when (io.bru.in.halt || io.vcore.undef) {
+  val vcoreUndef = if (p.enableVector) { io.vcore.get.undef } else { false.B }
+  when (io.bru.in.halt || vcoreUndef) {
     halted := true.B
   }
 
-  when (io.bru.in.fault || io.vcore.undef) {
+  when (io.bru.in.fault || vcoreUndef) {
     fault := true.B
   }
+  
 
   io.halted := halted
   io.fault  := fault
@@ -280,8 +288,10 @@
     io.counters.rfwriteCount +
     io.counters.storeCount +
     io.counters.branchCount +
-    io.counters.vrfwriteCount +
-    io.counters.vstoreCount
+    (if (p.enableVector) {
+      io.counters.vrfwriteCount.get +
+      io.counters.vstoreCount.get
+    } else { 0.U })
 
   when (io.bru.in.mode.valid) {
     mode := io.bru.in.mode.bits
diff --git a/hdl/chisel/src/kelvin/scalar/Decode.scala b/hdl/chisel/src/kelvin/scalar/Decode.scala
index 37a0c86..206c516 100644
--- a/hdl/chisel/src/kelvin/scalar/Decode.scala
+++ b/hdl/chisel/src/kelvin/scalar/Decode.scala
@@ -206,7 +206,9 @@
     val dvu = Decoupled(new DvuCmd)
 
     // Vector interface.
-    val vinst = Decoupled(new VInstCmd)
+    val vinst = if (p.enableVector) {
+      Some(Decoupled(new VInstCmd))
+    } else { None }
 
     // Branch status.
     val branchTaken = Input(Bool())
@@ -223,7 +225,7 @@
   val decodeEn = io.inst.valid && io.inst.ready && !io.branchTaken
 
   // The decode logic.
-  val d = DecodeInstruction(pipeline, io.inst.addr, io.inst.inst)
+  val d = DecodeInstruction(p, pipeline, io.inst.addr, io.inst.inst)
 
   val vldst = d.vld || d.vst
   val vldst_wb = vldst && io.inst.inst(28)
@@ -236,7 +238,9 @@
   val isCsrImm = d.isCsr() &&  io.inst.inst(14)
   val isCsrReg = d.isCsr() && !io.inst.inst(14)
 
-  val isVIop = (io.vinst.bits.op === VInstOp.VIOP)
+  val isVIop = if (p.enableVector) {
+    io.vinst.get.bits.op === VInstOp.VIOP
+  } else { false.B }
 
   val isVIopVs1 = isVIop
   val isVIopVs2 = isVIop && io.inst.inst(1,0) === 0.U  // exclude: .vv
@@ -267,8 +271,10 @@
 
 
   // Vector extension interlock.
-  val vinstEn = !(io.serializeIn.vinst || isVIop && io.serializeIn.brcond) &&
-                !(d.isVector() && !io.vinst.ready)
+  val vinstEn = if (p.enableVector) {
+      !(io.serializeIn.vinst || isVIop && io.serializeIn.brcond) &&
+      !(d.isVector() && !io.vinst.get.ready)
+  } else { false.B }
 
   // Fence interlock.
   // Input mactive used passthrough, prefer to avoid registers in Decode.
@@ -395,10 +401,12 @@
     d.getvl    -> MakeValid(true.B, VInstOp.GETVL),
     d.getmaxvl -> MakeValid(true.B, VInstOp.GETMAXVL),
   ))
-  io.vinst.valid := decodeEn && vinst.valid
-  io.vinst.bits.addr := rdAddr
-  io.vinst.bits.inst := io.inst.inst
-  io.vinst.bits.op := vinst.bits
+  if (p.enableVector) {
+    io.vinst.get.valid := decodeEn && vinst.valid
+    io.vinst.get.bits.addr := rdAddr
+    io.vinst.get.bits.inst := io.inst.inst
+    io.vinst.get.bits.op := vinst.bits
+  }
 
   // Scalar logging.
   io.slog := decodeEn && d.slog
@@ -476,7 +484,7 @@
 }
 
 object DecodeInstruction {
-  def apply(pipeline: Int, addr: UInt, op: UInt): DecodedInstruction = {
+  def apply(p: Parameters, pipeline: Int, addr: UInt, op: UInt): DecodedInstruction = {
     val d = Wire(new DecodedInstruction)
 
     // Immediates
@@ -557,27 +565,35 @@
     // Decode scalar log.
     val slog = DecodeBits(op, "01111_00_00000_xxxxx_0xx_00000_11101_11")
 
-    // Vector length.
-    d.getvl    := DecodeBits(op, "0001x_xx_xxxxx_xxxxx_000_xxxxx_11101_11") && op(26,25) =/= 3.U && (op(24,20) =/= 0.U || op(19,15) =/= 0.U)
-    d.getmaxvl := DecodeBits(op, "0001x_xx_00000_00000_000_xxxxx_11101_11") && op(26,25) =/= 3.U
+    if (p.enableVector) {
+      // Vector length.
+      d.getvl    := DecodeBits(op, "0001x_xx_xxxxx_xxxxx_000_xxxxx_11101_11") && op(26,25) =/= 3.U && (op(24,20) =/= 0.U || op(19,15) =/= 0.U)
+      d.getmaxvl := DecodeBits(op, "0001x_xx_00000_00000_000_xxxxx_11101_11") && op(26,25) =/= 3.U
 
-    // Vector load/store.
-    d.vld := DecodeBits(op, "000xxx_0xxxxx_xxxxx0_xx_xxxxxx_x_111_11")     // vld
+      // Vector load/store.
+      d.vld := DecodeBits(op, "000xxx_0xxxxx_xxxxx0_xx_xxxxxx_x_111_11")     // vld
 
-    d.vst := DecodeBits(op, "001xxx_0xxxxx_xxxxx0_xx_xxxxxx_x_111_11") ||  // vst
-             DecodeBits(op, "011xxx_0xxxxx_xxxxx0_xx_xxxxxx_x_111_11")     // vstq
+      d.vst := DecodeBits(op, "001xxx_0xxxxx_xxxxx0_xx_xxxxxx_x_111_11") ||  // vst
+               DecodeBits(op, "011xxx_0xxxxx_xxxxx0_xx_xxxxxx_x_111_11")     // vstq
 
-    // Convolution transfer accumulators to vregs. Also decodes acset/actr ops.
-    val vconv = DecodeBits(op, "010100_000000_000000_xx_xxxxxx_x_111_11")
+      // Convolution transfer accumulators to vregs. Also decodes acset/actr ops.
+      val vconv = DecodeBits(op, "010100_000000_000000_xx_xxxxxx_x_111_11")
 
-    // Duplicate
-    val vdup = DecodeBits(op, "01000x_0xxxxx_000000_xx_xxxxxx_x_111_11") && op(13,12) <= 2.U
-    val vdupi = vdup && op(26) === 0.U
+      // Duplicate
+      val vdup = DecodeBits(op, "01000x_0xxxxx_000000_xx_xxxxxx_x_111_11") && op(13,12) <= 2.U
+      val vdupi = vdup && op(26) === 0.U
 
-    // Vector instructions.
-    d.viop := op(0) === 0.U ||     // .vv .vx
-              op(1,0) === 1.U ||  // .vvv .vxv
-              vconv || vdupi
+      // Vector instructions.
+      d.viop := op(0) === 0.U ||     // .vv .vx
+                op(1,0) === 1.U ||  // .vvv .vxv
+                vconv || vdupi
+    } else {
+      d.getvl    := false.B
+      d.getmaxvl := false.B
+      d.vld      := false.B
+      d.vst      := false.B
+      d.viop     := false.B
+    }
 
     // [extensions] Core controls.
     d.ebreak := DecodeBits(op, "000000000001_00000_000_00000_11100_11")
diff --git a/hdl/chisel/src/kelvin/scalar/SCore.scala b/hdl/chisel/src/kelvin/scalar/SCore.scala
index 0c51a55..6c9c47e 100644
--- a/hdl/chisel/src/kelvin/scalar/SCore.scala
+++ b/hdl/chisel/src/kelvin/scalar/SCore.scala
@@ -36,9 +36,11 @@
     val ibus = new IBusIO(p)
     val dbus = new DBusIO(p)
     val ubus = new DBusIO(p)
-    val vldst = Output(Bool())
 
-    val vcore = Flipped(new VCoreIO(p))
+    val vldst = if (p.enableVector) { Some(Output(Bool())) } else { None }
+    val vcore = if (p.enableVector) {
+        Some(Flipped(new VCoreIO(p)))
+    } else { None }
 
     val iflush = new IFlushIO(p)
     val dflush = new DFlushIO(p)
@@ -127,7 +129,7 @@
     decode(i).io.scoreboard.regd := regfile.io.scoreboard.regd | scoreboard_spec(i)
   }
 
-  decode(0).io.mactive := io.vcore.mactive
+  decode(0).io.mactive := (if (p.enableVector) { io.vcore.get.mactive } else { false.B })
   for (i <- 1 until p.instructionLanes) {
     decode(i).io.mactive := false.B
   }
@@ -160,8 +162,10 @@
   csr.io.counters.rfwriteCount := regfile.io.rfwriteCount
   csr.io.counters.storeCount := lsu.io.storeCount
   csr.io.counters.branchCount := bru(0).io.taken.valid
-  csr.io.counters.vrfwriteCount := io.vcore.vrfwriteCount
-  csr.io.counters.vstoreCount := io.vcore.vstoreCount
+  if (p.enableVector) {
+    csr.io.counters.vrfwriteCount.get := io.vcore.get.vrfwriteCount
+    csr.io.counters.vstoreCount.get := io.vcore.get.vstoreCount
+  }
 
   // ---------------------------------------------------------------------------
   // Control Status Unit
@@ -170,7 +174,9 @@
   csr.io.req <> decode(0).io.csr
   csr.io.rs1 := regfile.io.readData(0)
 
-  csr.io.vcore.undef := io.vcore.undef
+  if (p.enableVector) {
+    csr.io.vcore.get.undef := io.vcore.get.undef
+  }
 
   // ---------------------------------------------------------------------------
   // Status
@@ -219,23 +225,35 @@
 
     regfile.io.writeData(i).valid := csr0Valid ||
                                      alu(i).io.rd.valid || bru(i).io.rd.valid ||
-                                     io.vcore.rd(i).valid
+                                     (if (p.enableVector) {
+                                        io.vcore.get.rd(i).valid
+                                      } else { false.B })
 
     regfile.io.writeData(i).addr :=
         MuxOR(csr0Valid, csr0Addr) |
         MuxOR(alu(i).io.rd.valid, alu(i).io.rd.addr) |
         MuxOR(bru(i).io.rd.valid, bru(i).io.rd.addr) |
-        MuxOR(io.vcore.rd(i).valid, io.vcore.rd(i).addr)
+        (if (p.enableVector) {
+           MuxOR(io.vcore.get.rd(i).valid, io.vcore.get.rd(i).addr)
+         } else { false.B })
+        
 
     regfile.io.writeData(i).data :=
         MuxOR(csr0Valid, csr0Data) |
         MuxOR(alu(i).io.rd.valid, alu(i).io.rd.data) |
         MuxOR(bru(i).io.rd.valid, bru(i).io.rd.data) |
-        MuxOR(io.vcore.rd(i).valid, io.vcore.rd(i).data)
+        (if (p.enableVector) {
+           MuxOR(io.vcore.get.rd(i).valid, io.vcore.get.rd(i).data)
+         } else { false.B })
 
-    assert((csr0Valid +&
-            alu(i).io.rd.valid +& bru(i).io.rd.valid +&
-            io.vcore.rd(i).valid) <= 1.U)
+    if (p.enableVector) {
+      assert((csr0Valid +&
+              alu(i).io.rd.valid +& bru(i).io.rd.valid +&
+              io.vcore.get.rd(i).valid) <= 1.U)
+    } else {
+      assert((csr0Valid +&
+              alu(i).io.rd.valid +& bru(i).io.rd.valid) <= 1.U)
+    }
   }
 
   val mluDvuOffset = p.instructionLanes
@@ -256,12 +274,9 @@
 
   // ---------------------------------------------------------------------------
   // Vector Extension
-  for (i <- 0 until p.instructionLanes) {
-    io.vcore.vinst(i) <> decode(i).io.vinst
-  }
-
-  for (i <- 0 until p.instructionLanes * 2) {
-    io.vcore.rs(i) := regfile.io.readData(i)
+  if (p.enableVector) {
+    io.vcore.get.vinst <> decode.map(_.io.vinst.get)
+    io.vcore.get.rs := regfile.io.readData
   }
 
   // ---------------------------------------------------------------------------
@@ -273,7 +288,9 @@
   io.dbus <> lsu.io.dbus
   io.ubus <> lsu.io.ubus
 
-  io.vldst := lsu.io.vldst
+  if (p.enableVector) {
+    io.vldst.get := lsu.io.vldst
+  }
 
   // ---------------------------------------------------------------------------
   // Scalar logging interface
diff --git a/tests/verilator_sim/kelvin/core_tb.cc b/tests/verilator_sim/kelvin/core_tb.cc
index 73396ab..9fff817 100644
--- a/tests/verilator_sim/kelvin/core_tb.cc
+++ b/tests/verilator_sim/kelvin/core_tb.cc
@@ -98,6 +98,7 @@
   sc_signal<sc_bv<32> > io_debug_inst2;
   sc_signal<sc_bv<32> > io_debug_inst3;
   sc_signal<sc_bv<32> > io_debug_cycles;
+#if 1
   sc_signal<bool> io_axi0_write_addr_ready;
   sc_signal<bool> io_axi0_write_addr_valid;
   sc_signal<sc_bv<32> > io_axi0_write_addr_bits_addr;
@@ -119,6 +120,7 @@
   sc_signal<sc_bv<2> > io_axi0_read_data_bits_resp;
   sc_signal<sc_bv<kUncId> > io_axi0_read_data_bits_id;
   sc_signal<sc_bv<kUncBits> > io_axi0_read_data_bits_data;
+#endif  // TODO: Disable if no VCore
   sc_signal<bool> io_axi1_write_addr_ready;
   sc_signal<bool> io_axi1_write_addr_valid;
   sc_signal<sc_bv<32> > io_axi1_write_addr_bits_addr;
@@ -230,6 +232,7 @@
 #define BINDAXI(a) \
   core.a(a);       \
   mif.a(a)
+#if 1
   BINDAXI(io_axi0_write_addr_ready);
   BINDAXI(io_axi0_write_addr_valid);
   BINDAXI(io_axi0_write_addr_bits_addr);
@@ -251,6 +254,7 @@
   BINDAXI(io_axi0_read_data_bits_resp);
   BINDAXI(io_axi0_read_data_bits_id);
   BINDAXI(io_axi0_read_data_bits_data);
+#endif  // TODO: Disable if no VCore
   BINDAXI(io_axi1_write_addr_ready);
   BINDAXI(io_axi1_write_addr_valid);
   BINDAXI(io_axi1_write_addr_bits_addr);