-
Notifications
You must be signed in to change notification settings - Fork 21
Tutorial page 2
Constructing combinational logic blocks in Chisel is fairly straightforward; when you declare a val in Scala, it creates a node that represents the data that it is assigned to. As long as the value is not assigned to be a register type (explained later), this tells the Chisel compiler to treat the value as wire. Thus any number of these values can be connected and manipulated to produce the value that we want.
Suppose we want to construct a single full adder. A full adder takes two inputs a and b, and a carry in cin and produces a sum and carry out cout. The Chisel source code for our full adder will look something like:
class FullAdder extends Module {
val io = new Bundle {
val a = UInt(INPUT, 1)
val b = UInt(INPUT, 1)
val cin = UInt(INPUT, 1)
val sum = UInt(OUTPUT, 1)
val cout = UInt(OUTPUT, 1)
}
// Generate the sum
val a_xor_b = io.a ^ io.b
io.sum := a_xor_b ^ io.cin
// Generate the carry
val a_and_b = io.a & io.b
val b_and_cin = io.b & io.cin
val a_and_cin = io.a & io.cin
io.cout := a_and_b | b_and_cin | a_and_cin
}
where cout is defined as a combinational function of inputs a, b, and cin.
You will notice that in order to access the input values from the io bundle, you need to first reference io since the input and output values belong to the io bundle. The |, &, and ˆ operators correspond to bitwise OR, AND, and XOR operations respectively.
If you don’t explicitly specify the width of a value in Chisel, the Chisel compiler will infer the bit width for you based on the inputs that define the value. Notice in the FullAdder definition, the widths for a_xor_b, a_and_b, b_and_cin, and a_and_cin are never specified anywhere. However, based on how the input is computed, Chisel will correctly infer each of these values are one bit wide since each of their inputs are the results of bitwise operations applied to one bit operands.
To generate Verilog code, move to the Learning Journey home directory and run:
test:runMain examples.Launcher FullAdder --backend-name=verilator
The Verilog is generated in FullAdder.v
found in ./test_run_dir/FullAdder
.
A quick inspection of the generated Verilog shows these values are indeed one bit wide:
module FullAdder(
input io_a,
input io_b,
input io_cin,
output io_sum,
output io_cout
);
wire a_xor_b;
wire _T_7;
wire a_and_b;
wire b_and_cin;
wire a_and_cin;
wire _T_8;
wire _T_9;
assign a_xor_b = io_a ^ io_b;
assign _T_7 = a_xor_b ^ io_cin;
assign a_and_b = io_a & io_b;
assign b_and_cin = io_b & io_cin;
assign a_and_cin = io_a & io_cin;
assign _T_8 = a_and_b | b_and_cin;
assign _T_9 = _T_8 | a_and_cin;
assign io_sum = _T_7;
assign io_cout = _T_9;
endmodule
Suppose we change the widths of the FullAdder to be 2 bits wide each instead such that the Chisel source now looks like:
class FullAdder extends Module {
val io = new Bundle {
val a = UInt(INPUT, 2)
val b = UInt(INPUT, 2)
val cin = UInt(INPUT, 2)
val sum = UInt(OUTPUT, 2)
val cout = UInt(OUTPUT, 2)
}
// Generate the sum
val a_xor_b = io.a ^ io.b
io.sum := a_xor_b ^ io.cin
// Generate the carry
val a_and_b = io.a & io.b
val b_and_cin = io.b & io.cin
val a_and_cin = io.a & io.cin
io.cout := a_and_b | b_and_cin | a_and_cin
}
As a result, the Chisel compiler should infer each of the intermediate values a_xor_b, a_and_b, b_and_cin, and a_and_cin are two bits wide. An inspection of the Verilog code correctly shows that Chisel inferred each of the intermediate wires in the calculation to be 2 bits wide.
module FullAdder(
input [1:0] io_a,
input [1:0] io_b,
input [1:0] io_cin,
output [1:0] io_sum,
output [1:0] io_cout
);
wire [1:0] a_xor_b;
wire [1:0] _T_7;
wire [1:0] a_and_b;
wire [1:0] b_and_cin;
wire [1:0] a_and_cin;
wire [1:0] _T_8;
wire [1:0] _T_9;
assign a_xor_b = io_a ^ io_b;
assign _T_7 = a_xor_b ^ io_cin;
assign a_and_b = io_a & io_b;
assign b_and_cin = io_b & io_cin;
assign a_and_cin = io_a & io_cin;
assign _T_8 = a_and_b | b_and_cin;
assign _T_9 = _T_8 | a_and_cin;
assign io_sum = _T_7;
assign io_cout = _T_9;
endmodule
Unlike Verilog, specifying a register in Chisel tells the compiler to actually generate a positive edge triggered register. In this section we explore how to instantiate registers in Chisel by constructing a shift register.
In Chisel, when you instantiate a register there are several ways to specify the connection of the input to a register. As shown in the GCD example, you can "declare" the register and assign what it’s input is connected to in a when... block or you can simply pass the value that the register is clocking as a parameter to the register.
If you choose to pass a next value to the register on construction using the next named parameter, it will clock the new value every cycle unconditionally:
// Clock the new register value on every cycle
val y = io.x
val z = Reg(next = y)
If we only want to update if certain conditions are met we use a when block to indicate that the registers are only updated when the condition is satisfied:
// Clock the new register value when the condition a > b
val x = Reg(UInt())
when (a > b) { x := y }
.elsewhen ( b > a) {x := z}
.otherwise { x := w}
It is important to note that when using the conditional method, the values getting assigned to the input of the register match the type and bitwidth of the register you declared. In the unconditional register assignment, you do not need to do this as Chisel will infer the type and width from the type and width of the input value.
The following sections show how these can be used to construct a shift register.
Suppose we want to construct a basic 4 bit shift register that takes a serial input in and generates a serial output out. For this first example we won’t worry about a parallel load signal and will assume the shift register is always enabled. We also will forget about the register reset signal.
If we instantiate and connect each of these 4 registers explicitly, our Chisel code will look something like:
class ShiftRegister extends Module {
val io = new Bundle {
val in = UInt(INPUT, 1)
val out = UInt(OUTPUT, 1)
}
val r0 = Reg(next = io.in)
val r1 = Reg(next = r0)
val r2 = Reg(next = r1)
val r3 = Reg(next = r2)
io.out := r3
}
If we take a look at the generated Verilog, we will see that Chisel did indeed map our design to a shift register. One thing to notice is that the clock signal and reset signals are implicitly attached to our design.
module ShiftRegister(
input clock,
input reset,
input io_in,
output io_out
);
reg r0;
reg r1;
reg r2;
reg r3;
assign io_out = r3;
always @(posedge clock) begin
r0 <= io_in;
r1 <= r0;
r2 <= r1;
r3 <= r2;
end
endmodule
As mentioned earlier, Chisel allows you to conditionally update a register (use an enable signal) using the when, .elsewhen, .otherwise block. Suppose we add an enable signal to our shift register, that allows us to control whether data is shift in and out on a given cycle depending on an enable input signal. The new shift register now looks like:
class ShiftRegister extends Module {
val io = new Bundle {
val in = UInt(INPUT, 1)
val enable = Bool(INPUT)
val out = UInt(OUTPUT, 1)
}
val r0 = Reg(UInt())
val r1 = Reg(UInt())
val r2 = Reg(UInt())
val r3 = Reg(UInt())
when (io.enable) {
r0 := io.in
r1 := r0
r2 := r1
r3 := r2
}
io.out := r3
}
Notice that it is not necessary to specify an .otherwise condition as Chisel will correctly infer that the old register value should be preserved otherwise.
###2.3.3 Register Reset
Chisel allows you to specify a synchronous reset to a certain value by specifying an additional parameter when you first declare them. In our shift register, let’s add a reset capability that resets all the register values to zero synchronously. To do this we need to provide our register declarations a little more information using the init parameter with what value we want on a synchronous reset:
class ShiftRegister extends Module {
val io = new Bundle {
val in = UInt(INPUT, 1)
val enable = Bool(INPUT)
val out = UInt(OUTPUT, 1)
}
// Register reset to zero
val r0 = Reg(init = UInt(0, width = 1))
val r1 = Reg(init = UInt(0, width = 1))
val r2 = Reg(init = UInt(0, width = 1))
val r3 = Reg(init = UInt(0, width = 1))
when (io.enable) {
r0 := io.in
r1 := r0
r2 := r1
r3 := r2
}
io.out := r3
}
Notice that reset value can actually be any value, simply replace the zeros and width to appropriate values.
Chisel also has an implict global reset signal that you can use in a when block. The reset signal is conveniently called reset and does not have to be declared. The shift register using this implict global reset now looks like:
class ShiftRegister extends Module {
val io = new Bundle {
val in = UInt(INPUT, 1)
val enable = Bool(INPUT)
val out = UInt(OUTPUT, 1)
}
val r0 = Reg(UInt())
val r1 = Reg(UInt())
val r2 = Reg(UInt())
val r3 = Reg(UInt())
when(reset) {
r0 := UInt(0)
r1 := UInt(0)
r2 := UInt(0)
r3 := UInt(0)
} .elsewhen(io.enable) {
r0 := io.in
r1 := r0
r2 := r1
r3 := r2
}
io.out := r3
}
The following exercises can be found in your $TUT_DIR/problems/ folder. You will find that some parts of the tutorial files have been completed for you and the section that you need to will need to complete is indicated in the file. The solutions to each of these exercises can be found in the $TUT_DIR/solutions/ folder.
The first tutorial problem is to write write a sequential circuit that sums in values. You can find the template in $LJHOME/src/main/scala/problems/Accumulator.scala
including a stubbed out version of the circuit:
class Accumulator extends Module {
val io = new Bundle {
val in = UInt(INPUT, 1)
val out = UInt(OUTPUT, 8)
}
// flush this out ...
io.out := UInt(0)
}
and a complete tester that confirms that you have successfully designed the circuit. Run
test:run-main problems.Launcher Accumulator
until your circuit passes the tests.