

Department of Electrical and Computer Engineering
Air University Islamabad
4th Semester

Computer Organization and Assembly Language Lab Complex Engineering Activity Instructors: Lab Engr. Ayesha Sadiq Dr. Muhammad Najam Dar

> Osama Anees Mirza 210286

Rayyan Munir 210301 Muneeb Khan 210276

09 Jun 2023

# Contents

| 1  | Objective:                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |                                                          |  |  |  |  |  |
|----|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|----------------------------------------------------------|--|--|--|--|--|
| 2  | Resources Required                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          | 3                                                        |  |  |  |  |  |
| 3  | Problem Statement:                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          |                                                          |  |  |  |  |  |
| 4  | Introduction:                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               |                                                          |  |  |  |  |  |
| 5  | The Format Of Our Instructions: 5.1 R-Type Instruction:                                                                                                                                                                                                                                                                                                                                                                                                                                                     | 3<br>4<br>4                                              |  |  |  |  |  |
| 6  | Procedure           6.1 Arithmetic Logic Unit (ALU)         6.1.1 Introduction:           6.1.2 Block Diagram:         6.1.2 Block Diagram:           6.2 Control Unit         6.2.1 Introduction:           6.2.2 Control Unit Flags:         6.2.3 Block Diagram:           6.3 ALU Control         6.3.1 ALUOp Table:           6.4 Memory Elements         6.4.1 Program Counter:           6.4.2 Instruction Memory:         6.4.3 Register File           6.4.4 Data Memory         6.4.4 Data Memory | 44<br>44<br>55<br>55<br>56<br>66<br>67<br>77<br>77       |  |  |  |  |  |
| 7  | Verilog Implementation           7.1 Program Counter         7.1.1 RTL Diagram           7.2 Program Counter Adder         7.2.1 RTL Diagram           7.3 Instruction Memory         7.3.1 RTL Diagram                                                                                                                                                                                                                                                                                                     | 16<br>18<br>18<br>20<br>20<br>22<br>22<br>23<br>23<br>24 |  |  |  |  |  |
| 8  |                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             | <b>26</b>                                                |  |  |  |  |  |
|    | 9.1       Output Wave Form                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  | 31                                                       |  |  |  |  |  |
| 10 | Conclusion                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  | 32                                                       |  |  |  |  |  |

## 1 Objective:

The aim of this intricate engineering activity is to introduce students to the concepts and processes involved in designing and implementing a datapath using Verilog, a hardware description language. A datapath is a crucial component of a computer's central processing unit (CPU) responsible for performing arithmetic and logical operations on data. The objective is to provide students with practical experience and a deep understanding of the inner workings of a CPU's datapath. This comprehensive engineering activity encompasses the following aspects:

- Familiarizing students with the principles and theory behind datapath design.
- Hands-on experience in implementing a datapath using Verilog.
- Gaining insight into the various stages and components of a CPU's datapath.
- Understanding the interaction and flow of data within the datapath.
- Practicing the design and implementation of arithmetic and logical operations.
- Exploring different control mechanisms and their integration with the datapath.

## 2 Resources Required

- Xilinx ISE Suite
- iverilog (Icarus Verilog)
- VS-Code
- A Computer (Not a broken one because you know its just a silicon rock at that point).

### 3 Problem Statement:

Students will be tasked with designing a 16-bit datapath for a simplified CPU that supports three instructions: ADD, SUB, and AND. The datapath should be able to perform these operations on 16-bit operands and produce a 16-bit result. The CPU has two general-purpose registers (R0 and R1) and a special-purpose register called the Accumulator (ACC) that holds the result of the most recent operation. The datapath should consist of the following components:

- 1. Register File
- 2. Arithmetic Logic Unit
- 3. Control Unit
- 4. Instruction Memory
- 5. Program Counter
- 6. Data Memory
- 7. Instruction decoder

#### 4 Introduction:

In this task we are going to design a 16 bit single cycle RICS core which supports MIPS like instructions. We just need to execute R-type instruction as stated in the problem statement.

### 5 The Format Of Our Instructions:

Before proceeding further we need to understand the format of our instruction. We are using 16 bits and only require to execute Register-type with Immediate-type instructions. The following table explains our instruction set:

### 5.1 R-Type Instruction:

| Register Type Instruction |     |       |            |                     |          |
|---------------------------|-----|-------|------------|---------------------|----------|
| Instruction               | op  | rs    | $_{ m rt}$ | $\operatorname{rd}$ | function |
| add                       | 000 | 3 bit | 3bit       | 3bit                | 0000     |
| sub                       | 000 | 3 bit | 3bit       | 3bit                | 0001     |
| and                       | 000 | 3 bit | 3bit       | 3bit                | 0010     |
| or                        | 000 | 3 bit | 3bit       | 3bit                | 0011     |

### 5.2 I-Type Instruction:

| Immediate Type Instruction |     |       |      |                       |  |  |
|----------------------------|-----|-------|------|-----------------------|--|--|
| Instruction op             |     | rs    | rt   | Shift Amount/Function |  |  |
| lw                         | 001 | 3 bit | 3bit | 0000000               |  |  |
| sw                         | 010 | 3 bit | 3bit | 0000000               |  |  |

These tables are modified versions of [1].

### 6 Procedure

## 6.1 Arithmetic Logic Unit (ALU)

#### 6.1.1 Introduction:

We will design an ALU that can perform a subset of the ALU operations of a full Processor ALU.



Figure 1: This image was taken from [2]

In this exercise, we will develop an ALU that will take two 2-inputs, A and B and is able to execute the following instructions:

| ALU Control | Instruction |
|-------------|-------------|
| 000 (add)   | lw,sw       |
| 000 (add)   | add         |
| 001 (add)   | sub         |
| 010 (and)   | and         |
| 011 (or)    | or          |

The ALU will generate a 16-bit output that we will call 'Result' and an additional 1-bit flag 'Zero' that will be set to 'logic-1' if all the bits of 'Result' are 0. The different operations will be selected by a 3-bit control signal called 'ALUControl' according to the following table.

For example, when the 'ALUControl' input is '011', the function Result = A or B should be calculated. It is easy to see that there are many values of 'ALUControl' for which no operation has been defined. It is not very important what the circuit does when 'ALUControl' has these values, since the 'Result' will simply be ignored in these cases. You can use this to your advantage to simplify the circuit. Right now, the described operations may look random, but once we learn more about the Instruction set architecture, these choices will make more sense.

### 6.1.2 Block Diagram:



Figure 2: Where N = 16. This image was taken from [2].

## 6.2 Control Unit

### 6.2.1 Introduction:

RISC-V consists of defining the following instruction formats: R-type, I-type, S-Type, B-Type, U-type, and J-type. R-type instructions operate on three registers. I-type, S-type and B-type instructions operate on two registers and a 12-bit immediate. U-type and J-type (jump) instructions operate on one 20-bit immediate. Here our only concern is of R-type and I-type.

### 6.2.2 Control Unit Flags:

Table 1: Control Unit Flags

| Instruction Type | op flag(input) | RegDst | ALUSrc | MemToReg | RegWrite | MemRead | MemWrite | ALUOp(2 bits) |
|------------------|----------------|--------|--------|----------|----------|---------|----------|---------------|
| R-Type           | 000            | 1      | 0      | 0        | 1        | 0       | 0        | 00            |
| lw               | 001            | 0      | 1      | 1        | 1        | 1       | 0        | 11            |
| SW               | 010            | 0      | 1      | 0        | 0        | 0       | 1        | 11            |

#### 6.2.3 Block Diagram:



Figure 3: Ignore The Non R-Type Flags

### 6.3 ALU Control

The main decoder computes most of the outputs from the opcode. It also determines a 2-bit ALUOp signal. The ALU decoder uses this ALUOp signal in conjunction with the funct field and opcode bit to compute ALUControl. The meaning of the ALUOp signal is given in Table below:

#### 6.3.1 ALUOp Table:

| ALUOp | Meaning                             |
|-------|-------------------------------------|
| 00    | add                                 |
| 01    | subtract                            |
| 10    | look at funct fields and opcode bit |
| 11    | N/A                                 |

Table below is a truth table for the ALU decoder. The logic of ALUControl was covered in the above chapter. When ALUOp is 00 or 01, the ALU should add or subtract, respectively. When ALUOp is 10, the decoder examines the function fields and operand bit to determine the ALUControl. The control signals for each instruction were described as we built the datapath.

## 6.4 Memory Elements

These include:

- 1. Program Counter
- 2. Instruction Memory
- 3. Register File
- 4. Data Memory



Figure 4: This image was taken from [2]

#### 6.4.1 Program Counter:

The Program Counter (PC) is a fundamental component in computer architecture responsible for storing the memory address of the next instruction to be executed. In the case of a 16-bit Program Counter, it can store a 16-bit memory address, allowing it to access up to 65,536 memory locations.

The Program Counter works in conjunction with the Instruction Fetch stage of the processor's pipeline. During each clock cycle, the PC increments by one to point to the next memory location. It fetches the instruction stored at that memory address and sends it to the Instruction Decoder for further processing.

When an instruction requires a branch or jump, the Program Counter is modified to point to the new memory address indicated by the branch instruction. This allows the processor to change the sequence of instruction execution and alter the program flow.

The Program Counter is a vital component for sequential instruction execution, ensuring the processor fetches and executes instructions in the correct order. It plays a crucial role in maintaining program flow and enabling the processor to follow the control flow of the program being executed.

#### 6.4.2 Instruction Memory:

The Instruction Memory is a crucial component in computer architecture responsible for storing the program instructions that the processor executes. In the case of a 16-bit Instruction Memory, it can store and provide access to 16-bit instructions.

The Instruction Memory works in conjunction with the Program Counter (PC) and the Instruction Decoder. The PC holds the memory address of the next instruction to be fetched, and the Instruction Memory fetches the instruction from the corresponding memory location.

The Instruction Memory operates by receiving the memory address from the PC and retrieving the instruction stored at that address. It then sends the instruction to the Instruction Decoder, which interprets the instruction and initiates the necessary operations.

The Instruction Memory is typically implemented as a read-only memory (ROM) or as a portion of the main memory in a computer system. It is initialized with the program instructions prior to program execution and remains constant during runtime.

The Instruction Memory's primary function is to provide the processor with the instructions needed to execute a program. It ensures that the processor fetches the correct instructions in the correct order, enabling the execution of complex programs and algorithms.

#### 6.4.3 Register File

The Register File is a component that stores and provides access to a set of 16-bit general-purpose registers in a computer architecture. It allows the processor to quickly read and write data during instruction execution, improving performance and efficiency.

#### 6.4.4 Data Memory

The data memory has a single read/write port. If the write enable, WE, is 1, it writes data WD into address A on the rising edge of the clock. If the write enable is 0, it reads address A onto RD.

The instruction memory, register file, and data memory are all read combinationally. In other words, if the address changes, the new data appears at RD after some propagation delay; no clock is involved. They are written only on

the rising edge of the clock. In this fashion, the state of the system is changed only at the clock edge. The address, data, and write enable must be set up sometime before the clock edge and must remain stable until a hold time after the clock edge. Because the state elements change their state only on the rising edge of the clock, they are synchronous sequential circuits.

## 7 Verilog Implementation

We used Xilinx along with Icarus Verilog and VS-Code to code, test and make its RTL. We have separated all the individual components into separate files as it is much easier to detect errors and maintain the code base.

## 7.1 Program Counter

### 7.1.1 RTL Diagram



Figure 5: RTL Block Diagram



Figure 6: Program Counter

```
Code:
module PC(PC_NEXT,pc,rst,clk);
    //Declaring Inputs:
    input [15:0] PC_NEXT;
          input clk,rst;
          //Declaring Outputs:
output reg [15:0] pc;
          always @(posedge clk) begin //On every Positive Edge of the clock do:
    if (rst == 1'b0)// Active Low Values.
10
                              pc <= 16'b000000000000000000000;</pre>
11
                       end
12
13
                       begin
14
                             pc <= PC_NEXT;</pre>
15
16
17
          end
18 endmodule
```

## 7.2 Program Counter Adder

## 7.2.1 RTL Diagram



Figure 7: RTL Block Diagram



Figure 8: Program Counter Adder

```
Code:
module PC_Adder (OLD_PC,NEW_PC,Four16Bit);

//Declaring Inputs:
input [15:0]OLD_PC,Four16Bit;

//Declaring Outputs:
output [15:0] NEW_PC;

//Assigning Outputs:
assign NEW_PC = OLD_PC + Four16Bit;

endmodule
```

## 7.3 Instruction Memory

## 7.3.1 RTL Diagram



Figure 9: RTL Block Diagram

# Instr\_Mem:1 Mmux\_RD1 Mram mem1 Data0(15:0) A(15:0) addrA(9:0) doA(15:0) Result(15:0) RD(15:0) diA(15:0) Data1(15:0) Sel(0) Mram\_mem1 Mmux\_RD1 gnd XST\_GND inv rst rst\_INV\_1\_o1

Figure 10: Instruction Memory

Instr Mem

```
2 input rst;
3 input [15:0]A;
4 output [15:0]RD;
6 reg [15:0] mem [1023:0];
8 assign RD = (~rst) ? {16{1'b0}} : mem[A[15:2]];
10 // initial begin
11 // $readmemh("/home/baymax/Air-Uni-EE/32Bit/RISCV_Single_Cycle_Core-v.1.0.0/src/memfile.hex",mem);
12 // end
13
14
15
16 initial begin
     mem[0] = 16'b0000011010001000; // add $1,$2,$3
mem[1] = 16'b0001011010001000; // sub $1,$2,$3
17
     mem[2] = 16'b0010011010001000; // and $1,$2,$3
19
     // mem[3] = 16'b0001011010001000;

// mem[4] = 16'b00001011010001000;

// mem[5] = 16'b0001011010001000;
20
21
22
23
24 end
25
26 endmodule
```

## 7.4 Register File

## 7.4.1 RTL Diagram



Figure 11: RTL Block Diagram



Figure 12: Register File

```
Code:
module Reg_File (A1,A2,A3,WD3,WE3,clk,rst,RD1,RD2);
    //NOTES:
      // WD3 = Write Data
      // WE3 = Write Enable (Comming From Control Unit's RegWrite Output)
      //Declaring Inputs:
      input [2:0] A1,A2,A3;
      input [15:0] WD3;
      input clk,rst,WE3;
10
      //Declaring Outputs:
11
      output [15:0] RD1,RD2;
13
      //Creation Of The Memory:
14
      reg[15:0] Registers [15:0];
16
      17
18
      assign RD2 = (!rst) ? 16'b0000000000000 : Registers[A2];
19
20
21
      //Write Functionality:
      always @(posedge clk) begin// Positive Edge Of The Clock
22
          if (WE3)
23
              begin
24
                  Registers[A3] <= WD3;</pre>
25
26
27
28
      //We can Initialize Data Of Registers From Here!
30
```

```
initial begin
Registers[0] = 16'b00000000000000;// $zero
Registers[1] = 16'b0000000000000000;
Registers[2] = 16'b000000000000000;
Registers[3] = 16'b000000000000001;
Registers[4] = 16'b0000000000000000;
rend
end
```

## 7.5 Control Unit

## 7.5.1 RTL Diagram



Figure 13: RTL Block Diagram



Figure 14: Control Unit

```
Code:
  module control_unit (op,RegWrite,MemWrite,ALUSrc,ALUOp,RegDst,MemToReg,MemRead);
      //Inputs / Outputs declaration
      input [2:0] op;
      output RegWrite,MemWrite,ALUSrc,RegDst,MemToReg,MemRead;
      output [1:0] ALUOp;
      //Assigning Outputs:
9
      // NOTES:
10
11
      11
      // |Instruction | OP | RegDst | ALUSrc | MemToReg | RegWrite | MemRead | MemWrite | ALUOp |
      // | R-Type
                     | 000 |
                            1 |
                                       0 |
                                                0
                                                        1
                                                                   0
                                                                            0
                                                                                      00
13
                     | 001 |
                               0
      // | Load Word
                                       1
                                                1
                                                          1
                                                                             0
                                                                                      11
14
                                 -1
      // |Store Word | 010 |
                              0
                                                0
                                                          0
                                                                   1
                                                                            0
15
                                       1
                                                                                     11
      assign RegWrite = ((op == 3'b000) | (op == 3'b001)) ? 1'b1 : 1'b0;// if (op == R-type | op == lw)
17
18
      assign RegDst = (op == 3'b000) ? 1'b1 : 1'b0;// if (op == R-type)
19
20
      21
22
      assign MemToReg = (op == 3'b001) ? 1'b1 : 1'b0;// if (op == lw)
23
24
      assign MemRead = ((op == 3'b001) | (op == 3'b010)) ? 1'b1 : 1'b0;// if (op == lw | op == sw)
25
26
      assign MemWrite = (op == 3'b010) ? 1'b1 : 1'b0;// if (op == sw)
27
28
      assign ALUOp = ((op == 3'b001) | (op == 3'b010)) ? 2'b11 : 2'b00;// if (op == lw | op == sw)
29
30
```

## **7.6** ALU

## 7.6.1 RTL Diagram



Figure 15: RTL Block Diagram



Figure 16: ALU

```
Code: module alu (A,B,ALUControl,Result,Negative,Carry,Zero);
       //declaring inputs
       input [15:0] A,B;
       input [2:0] ALUControl;
       //declaring Outputs
       output [15:0] Result;
       output Negative, Carry, Zero;
10
       //declaring intermediate wires
11
       //AND or OR:
12
13
       wire [15:0] a_and_b;
       wire [15:0] a_or_b;
14
15
       //Addition or Subtraction:
16
       wire [15:0] not_b;
wire [15:0] sum;
17
18
       wire cout;
19
       //Mux
20
       wire [15:0] mux_1; //2's Complement Or Not
21
       wire [15:0] mux_2;
22
23
       //logic Design Outputs
24
       //AND Operation
25
       assign a_and_b = A & B;
26
27
       //OR Operation (Additional)
28
29
       assign a_or_b = A | B;
30
```

```
//Tenary Operator
31
       //assign name = (Condition) ? True Value : False Value;
32
       assign mux_1 = (ALUControl[0] == 2'b0) ? B : not_b; //2's complement
33
34
35
      //Addition or Subtraction:
      assign not_b = ~B; // 2's complement of B
36
      assign {cout,sum} = A + mux_1 + ALUControl[0];
37
38
39
      assign mux_2 = (ALUControl[1:0] == 2'b00) ? sum :
40
                      (ALUControl[1:0] == 2'b01) ? sum :
                      (ALUControl[1:0] == 2'b10) ? a_and_b : a_or_b;
42
43
44
       // assign mux_2 = (ALUControl[1] == 1'b0) ? sum :
                        (ALUControl[0] == 1'b0) ? a_and_b : a_or_b;
45
46
      //Result:
47
      assign Result = mux_2;
48
      //Flag Assignments:
50
      assign Zero = &(~Result); //Zero Flag
51
52
      assign Negative = Result[15]; //Negative Flag
      assign Carry = cout & (~ALUControl[1]); //Carry Flag
53
54
55 endmodule
```

### 7.7 ALU Control

## 7.7.1 RTL Diagram



Figure 17: RTL Block Diagram



Figure 18: ALU Control

```
//Declaring Inputs:
       input [1:0] alu_op;
      input [3:0] Fun;
      //Declaring Outputs:
      output [2:0] ALUControl;
      //Intermediate Wire
      wire [5:0] ALUControlIn;
10
      assign ALUControlIn = {alu_op,Fun}; //Contatinate. Same Pattern {Fun,alu_op}!={alu_op,Fun};
11
      //Tenary Operator
13
      assign ALUControl = (ALUControlIn == 6'b11xxxx) ? 3'b000: // lw,sw I-type
14
                            (ALUControlIn == 6'b000000) ? 3'b000: // Add R-type
15
                            (ALUControlIn == 6'b000001) ? 3'b001: // Sub R-type
16
                           (ALUControlIn == 6'b000010) ? 3'b010: // AND R-type (ALUControlIn == 6'b000011) ? 3'b011: // Or R-type
17
18
                           3'b000;
19
20 endmodule
```

## 7.8 Sign Extender

### 7.8.1 RTL Diagram



Figure 19: Sign Extender

```
Code:
    module Sign_Extend (In_Str,Extended_Out);

//Declaring Inputs:
    input [15:0] In_Str;

//Declaring Outputs:
    output [15:0] Extended_Out;

//Assigning Outputs:
    assign Extended_Out = (In_Str[9]) ? {{9{1'b1}},In_Str[15:9]};

endmodule
```

## 7.9 Mux 2X1

## 7.9.1 RTL Diagram



Figure 20: 2X1 Multiplexer

```
Code:
    module Mux (In1,In2,Selection,Out);

//Declaring Inputs:
    input [15:0] In1,In2;
    input Selection;

//Declaring Output:
    output [15:0] Out;

//Assigning Output:
    assign Out = (~Selection) ? In1 : In2;
endmodule
```

## 7.10 Data Memory

## 7.10.1 RTL Diagram



Figure 21: RTL Block Diagram



Figure 22: Data Memory

```
Code:
module Data_Mem (A,WD,WE,RE,clk,RD,rst);
//Declaring Inputs:
       input [15:0] A,WD;
       input clk,WE,RE,rst;
       //Declaruing Outputs:
       output [15:0] RD;
       //Creation OF Memory:
10
       reg[15:0] Data_MEM [255:0];
11
12
13
14
       //Read Functionality:
15
       assign RD = ((WE == 1'b0) & (WD == 1'b1) & (rst)) ? Data_MEM[A] : 16'b0;
16
17
       //Write Functionality:
18
       always @(posedge clk) begin
19
           if(WE)
20
21
                   Data_MEM[A] <= WD;</pre>
22
               end
23
24
25
       // We can Initialize Data Of The Memory From Here!
26
27
       // initial begin
       // Mem[9] = 16'b000000000000000;
28
       // end
29
30
```

# 8 Total Combined Data Path

## 8.1 Single Cycle MIPS

## 8.1.1 RTL Diagram



Figure 23: Block Diagram



Figure 24: A Closer Look



Figure 25: A More Better Look!



Figure 26: A More More Better Look!



Figure 27: A More More More Better Look!

```
Code:
   include "/home/baymax/Air-Uni-EE/Verilog/Program_Counter.v"
   include "/home/baymax/Air-Uni-EE/Verilog/Instruction_Memory.v"
   `include "/home/baymax/Air-Uni-EE/Verilog/Register_File.v"
4 `include "/home/baymax/Air-Uni-EE/Verilog/Sign_Extender.v"
   include "/home/baymax/Air-Uni-EE/Verilog/ALU.v"
  include "/home/baymax/Air-Uni-EE/Verilog/Control_Unit.v"
7 include "/home/baymax/Air-Uni-EE/Verilog/ALU_Control.v"
   include "/home/baymax/Air-Uni-EE/Verilog/Data_Memory.v"
   include "/home/baymax/Air-Uni-EE/Verilog/PC_Adder.v"
include "/home/baymax/Air-Uni-EE/Verilog/Mux.v"
module Single_Cycle_Top (alu_Carry_Out,alu_Negative_Out,alu_Zero_Out,random,clk,rst,clk_out,rst_out,Reg_Write_Out,
       MemWrite_Out,ALUSrc_Out,RegDst_Out,MemToReg_Out,MemRead_Out,ALUOp_Out);
       //Declaring Inputs:
12
      input rst,clk;
13
14
      wire[15:0] PC_Top,RD_Instr,RD1_Top,RD2_Top,SignExt_Top,ALU_RESULT;// For Connecting RD (From Instruction Memory) To
        Register File
16
       //Control Unit's Wires:
      wire RegWrite,MemWrite,ALUSrc,RegDst,MemToReg,MemRead;
17
      wire [1:0] ALUOp;
18
       //ALU_Control's Wires:
19
      wire [2:0] ALUControl_Top;
20
      //DataMemory:
21
      wire [15:0] ReadData;
22
      //PC's Wires:
23
24
      wire [15:0] NEW_PC;
      //Mux_Register_To_ALU Result:
25
      wire [15:0] Mux_Register_To_ALU_Result;
26
       wire [15:0] Mux_Data_Memory_To_Register_File_Write_Back_Result;
27
28
```

```
//Declaring Outputs:
29
       output clk_out,rst_out,Reg_Write_Out,MemWrite_Out,ALUSrc_Out,RegDst_Out,MemToReg_Out,MemRead_Out,alu_Negative_Out,
30
        alu_Zero_Out,alu_Carry_Out;
       output [1:0] ALUOp_Out;
31
       output [2:0] random;
32
       //Assigning The Outputs:
33
       assign clk_out = clk;
34
35
       assign rst_out = rst;
       assign Reg_Write_Out = Reg_Write_Out;
36
       assign MemWrite_Out = MemWrite;
37
38
       assign ALUSrc_Out = ALUSrc;
       assign RegDst_Out = RegDst;
39
       assign MemToReg_Out = MemToReg;
40
41
       assign MemRead_Out = MemRead;
       assign ALUOp_Out = ALUOp;
42
       assign random = RD_Instr[8:6];
43
44
       //assign alu_Zero_Out = ;
45
46
        //assign alu_Negative_Out = ;
47
       //assign alu_Carry_Out = ;
48
49
       PC PC(
            .clk(clk),
50
51
            .rst(rst),
            .pc(PC_Top),
52
            .PC_NEXT(NEW_PC)
53
       );
54
55
56
57
       PC_Adder PC_Adder(
            .OLD_PC(PC_Top),
58
            .NEW_PC(NEW_PC),
59
60
            .Four16Bit(16'b0000000000000100)
61
62
63
       Instr_Mem Instruction_Memory(
64
            .rst(rst),
65
            .A(PC_Top),
66
            .RD(RD_Instr)
67
68
69
70
71
       control_unit Control_Unit(
            .op(RD_Instr[2:0]),
72
73
            .RegWrite(RegWrite),
            .MemWrite(MemWrite),
74
75
            .ALUSrc(ALUSrc),
            .ALUOp(ALUOp),
76
            .RegDst(RegDst),
77
78
            . {\tt MemToReg(MemToReg)}\,,
79
            .MemRead(MemRead)
       );
80
81
       alu_control ALU_Control (
82
            .alu_op(ALUOp),
83
            .ALUControl(ALUControl_Top),
84
            .Fun(RD_Instr[15:12])
85
86
       );
87
88
89
90
       Reg_File Register_File(
91
            .A1(RD_Instr[5:3]),
92
            .A2(RD_Instr[8:6]),
93
94
            .A3(RD_Instr[11:9]),
            .WD3(Mux_Data_Memory_To_Register_File_Write_Back_Result),
95
            .WE3(RegWrite),
96
97
            .clk(clk),
            .rst(rst),
98
            .RD1(RD1_Top),
99
100
            .RD2(RD2_Top)
       ):
102
```

```
104
       Sign_Extend Sign_Extend(
105
106
            .In_Str(RD_Instr),
            .Extended_Out(SignExt_Top)
107
       );
108
109
       Mux Mux_Register_To_ALU(
110
111
            .In1(RD2_Top),
            .In2(SignExt_Top),
112
            .Selection(ALUSrc),
113
            .Out(Mux_Register_To_ALU_Result)
114
115
116
117
       alu ALU(
            .A(RD1_Top),
118
            .B(Mux_Register_To_ALU_Result),
119
            .ALUControl(ALUControl_Top),
120
            .Result(ALU_RESULT),
121
122
            .Negative(alu_Negative_Out),
123
            .Carry(alu_Carry_Out),
            .Zero(alu_Zero_Out)
124
125
126
       Data_Mem Data_Memory (
127
            .A(ALU_RESULT),
128
            .WD(RD2_Top),
129
            .WE(MemWrite),
130
            .RE(MemRead),
131
            .clk(clk),
132
133
            .RD(ReadData),
            .rst(rst)
134
       );
135
136
       Mux Mux_Data_Memory_To_Register_File_Write_Back(
137
138
            .In1(ALU_RESULT),
            .In2(ReadData),
139
            .Selection(MemToReg),
140
141
            .Out(Mux_Data_Memory_To_Register_File_Write_Back_Result)
       );
142
143 endmodule
```

## 9 Testing

### 9.1 Output Wave Form





### 9.2 Test Bench

```
Code:

'include "/home/baymax/Air-Uni-EE/Verilog/Single_Cycle_Top.v"

module Single_Cycle_Top_tb ();

reg clk=1'b1,rst;
```

```
4
       Single_Cycle_Top Single_Cycle_Top (
5
            .clk(clk),
6
            .rst(rst)
7
8
       );
9
       always
            begin
11
                clk = ~clk;
                #100;
            end
       initial begin
14
           rst <= 1'b0;
16
            #150;
17
           rst <=1'b1;
18
19
            #450;
            $finish;
20
       end
       initial begin
            $dumpfile("Single_Cycle.vcd");
23
24
            $dumpvars(0);
25
   endmodule
26
```

#### The following Instructions were given:

```
initial begin
mem[0] = 16'b0000011010001000; // add $1,$2,$3
mem[1] = 16'b0001011010001000; // sub $1,$2,$3
mem[2] = 16'b0010011010001000; // and $1,$2,$3
end
```

#### 10 Conclusion

In conclusion, we have completed this complex engineering activity with the objective of gaining practical experience in designing and implementing a datapath using Verilog, a hardware description language. Our focus throughout this activity was to develop a deep understanding of the inner workings of a CPU's datapath and its various components.

Throughout this activity, we have worked with key elements of the datapath, including the Register File, Arithmetic Logic Unit (ALU), Control Unit, Instruction Memory, Program Counter (PC), Data Memory, and Instruction Decode. By actively engaging with these components, we have gained hands-on experience in designing and implementing the fundamental building blocks of a CPU.

The Register File serves as a storage space for data manipulation, while the ALU performs arithmetic and logical operations on that data. The Control Unit plays a crucial role in coordinating the overall operation of the datapath, ensuring proper sequencing and control flow. The Instruction Memory and Program Counter work together to fetch and execute instructions, while the Data Memory provides temporary storage for data during program execution. Lastly, the Instruction Decode phase translates the fetched instructions into control signals for the datapath components.

Through this comprehensive engineering activity, we have developed a thorough understanding of how the datapath functions and how its various components interact. This knowledge will empower us to tackle future projects and challenges in the field of computer engineering, as we can apply our expertise in designing and implementing CPU datapaths using Verilog.

## References

- [1] FPGA4Students, "Verilog code for 16-bit single cycle mips processor," https://www.fpga4student.com/2017/01/verilog-code-for-single-cycle-MIPS-processor.html.
- $[2] \begin{tabular}{ll} MuhammadShameelAnsari1999 & and & HamzaShabbir517, & "Riscv\_single\_cycle\_core," & https://github.com/merldsu/RISCV\_Single\_Cycle\_Core & https://www.youtube.com/watch?v=BVvDHhG0RoA&list=PL5AmAh9QoSK7Fwk9vOJu-3VqBng\_HjGFc. & https://www.youtube.com/watch?v=BVvDHhG0RoA&list=PL5AmAh9Q$