

**NATIONAL UNIVERSITY OF SCIENCE AND  
TECHNALOGY**

**DEPARTMENT OF COMPUTER AND SOFTWARE  
ENGINEERING**

**Lab#9: J Type Instruction datapath and Control unit Including  
LW & SW Control Unit**

**Lab Report # 09**

**Course Instructor: Dr Shahid Ismail**

**Lab Instructor: Usama Shoukat**

| Sr. No. | Student's Name | CMS ID |
|---------|----------------|--------|
| 01      | SHAZIL         | 532263 |

**Due Date: 18-Nov-2025**  
**Submission Date: 17-Nov-2025**

## TABLE OF CONTENT

- 1) Introduction
- 2) Objectives
- 3) Software or Equipments
- 4) Lab tasks
- 5) Outputs
- 6) Conclusion

### **INTRODUCTION:**

#### **Jumps**

There are two jump instructions namely

1. jalr (I-Type)
2. jal (J-Type)

The encoding of J-Type is given in the figure below



A list of jump instructions is provided in the table below

| op            | funct3 | funct7 | Type | Instruction       | Description            | Operation                            |
|---------------|--------|--------|------|-------------------|------------------------|--------------------------------------|
| 1100111 (103) | 000    | -      | I    | jalr rd, rsl, imm | jump and link register | PC = rsl + SignExt(imm), rd = PC + 4 |
| 1101111 (111) | -      | -      | J    | jal rd, label     | jump and link          | PC = JTA, rd = PC + 4                |

The **jal** instruction jumps to the address specified in the immediate field, while the **jalr** uses the addresses stored in **rsl** along with immediate values to make longer jumps

### **OBJECTIVE:**

The objective of this lab is to Extend the previously designed Datapath to include loads and store controls and jump instructions datapath.

### **SOFTWARE\TOOL USED:**

Xilinx Vivado

## Task

Extend the design from previous task to include the jump instructions. After extending the datapath your design may look along the lines of the following figure. Secondly do add functionality of LW, SW and Jump Control signals.



## Explanation:

### OUTPUT:

#### PC MODULE:

```

22
23 module PC #(parameter N = 32)
24 (
25     input logic clk,
26     input logic reset,
27     output logic [N-1:0] add
28 );
29 always_ff@(posedge clk)
30 begin
31 if(reset == 1'b1)
32     add <= 0;
33 else
34     add <= add+4 ;
35 end
36 endmodule
37

```

## INSTRUCTION MEMORY

```

20 ///////////////////////////////////////////////////////////////////
21 module InsMemory #(parameter length = 116 , width = 32 , N = 8)
22   (
23     input logic [width-1:0] addr,
24     output logic [width-1:0] dataR
25   );
26   logic [N-1:0] InsMem [0:length-1];
27 initial
28 begin
29   $readmemh("IMdata.mem" , InsMem);
30 end
31
32 always_comb
33 begin
34   dataR = {InsMem[addr+3],InsMem[addr+2],InsMem[addr+1],InsMem[addr]};
35 end
36 endmodule
37

```

## DATA MEMORY:

```

///////////////////////////////////////////////////////////////////
module data_mem #(
  parameter DEPTH = 1024
) (
  input  logic      clk,
  input  logic      mem_read,
  input  logic      mem_write,
  input  logic [2:0] funct3,
  input  logic [31:0] addr,
  input  logic [31:0] dataW,
  output logic [31:0] dataR
);

  logic [7:0] mem [0:DEPTH-1];

  initial begin
    $readmemh("DataMemory.mem", mem);
  end

  logic [31:0] word;
  logic [31:0] write_word;
  logic [31:0] result;

```

```

initial begin
    $readmemh("DataMemory.mem", mem);
end

logic [31:0] word;
logic [31:0] write_word;
logic [31:0] result;

always_comb begin
    word = { mem[addr+3], mem[addr+2], mem[addr+1], mem[addr] };

    case(func3)
        3'b000: result = {{24{mem[addr][7]}}, mem[addr]};
        3'b100: result = {24'b0, mem[addr]};

        3'b001: result = {{16{word[15]}}, word[15:0]};
        3'b101: result = {16'b0, word[15:0]};

        3'b010: result = word;

        default: result = 32'h0;
    endcase

    if(mem_read)
        dataR = result;
    else
        dataR = 32'h0;
end

```

```

always_ff @ (posedge clk)
begin
    if(mem_write)
        begin
            case(func3)
                3'b000: mem[addr] <= dataW[7:0];

                3'b001: begin
                    mem[addr] <= dataW[7:0];
                    mem[addr+1] <= dataW[15:8];
                end

                3'b010: begin
                    mem[addr] <= dataW[7:0];
                    mem[addr+1] <= dataW[15:8];
                    mem[addr+2] <= dataW[23:16];
                    mem[addr+3] <= dataW[31:24];
                end
            endcase
        end
    end

    final begin
        $writememh("DataMemory.mem", mem);
    end
endmodule

```

## REGISTER FILE:

```

0 module Reg_file#(
1   parameter DATA_WIDTH = 32,
2   parameter NUM_REGS  = 32,
3   parameter e =6
4 )(
5   input logic clk,
6   input logic we,
7   input logic [e-1:0]rs1,
8   input logic [e-1:0]rs2,
9   input logic [e-1:0]rsw,
10  input logic [DATA_WIDTH-1:0]dataw,
11  output logic [DATA_WIDTH-1:0]data1,
12  output logic [DATA_WIDTH-1:0]data2
13 );
14   logic [DATA_WIDTH-1:0] regfile [0:NUM_REGS-1];
15   initial
16   begin
17     $readmemh("rfdata.mem" , regfile);
18   end
19
20   always_ff@(posedge clk)
21   begin
22     if(we && rsw !=0)
23     begin
24       regfile[rsw] <= dataw;
25       $display("Time=%0t | Wrote %h to regfile[%0d]", $time, dataw, rsw);
26     end
27   end
28
29   assign data1 = regfile[rs1];
30   assign data2 = regfile[rs2];
31
32 endmodule

```

## IMMEDIATE GENERATOR:

```

23 module ImmGen(
24   input logic [31:0] instr,
25   output logic [31:0] imm
26 );
27   logic [11:0] imm_I;
28   logic [11:0] imm_S;
29   logic [20:0] imm_J;
30   assign imm_I = instr[31:20];
31   assign imm_S = {instr[31:25], instr[11:7]};
32   assign imm_J = {instr[31], instr[19:12], instr[20], instr[30:21], 1'b0};|
33 always_comb begin
34   case (instr[6:0])
35     7'b0000011: imm = {{20{imm_I[11]}}, imm_I};      // LOAD
36     7'b0100011: imm = {{20{imm_S[11]}}, imm_S};      // STORE
37     7'b1101111: imm = {{11{imm_J[20]}}, imm_J};      // JAL
38     7'b1100111: imm = {{20{imm_I[11]}}, imm_I};      // JALR
39     default:    imm = {{20{imm_I[11]}}, imm_I};      // I-type
40   endcase
41 end

```

## ALU MODULE:

```
module ALU #(  
    parameter WIDTH = 32  
) (  
    input  logic [WIDTH-1:0] A,  
    input  logic [WIDTH-1:0] B,  
    input  logic [3:0] opcode,  
    output logic [WIDTH-1:0] result  
  
    always_comb  
    begin  
        case(opcode)  
            4'b0000:  
                begin  
                    result = A & B;  
                end  
            4'b0001:  
                begin  
                    result = A | B;  
                end  
            4'b0010:  
                begin  
                    result = A + B;  
                end  
            4'b0011:  
                begin  
                    result = A - B;  
                end  
        endcase  
  
    end  
endmodule
```

## **CONTROL UNIT:**

```
20 //////////////////////////////////////////////////////////////////
21 module control_unit(
22     input logic [6:0]opcode,
23     output logic regwrite,
24     output logic alusrc,
25     output logic memread,
26     output logic memwrite,
27     output logic memtoreg,
28     output logic branch,
29     output logic [1:0] aluop
30 );
31 always_comb
32 begin
33     regwrite = 1'b0;
34     alusrc = 1'b0;
35     memread = 1'b0;
36     memwrite = 1'b0;
37     memtoreg = 1'b0;
38     branch = 1'b0;
39     aluop = 2'b00;
40
41 case(opcode)
42    7'b00000011: //lw
43    begin
44        regwrite = 1'bl;
45        alusrc = 1'bl;
46        memread = 1'bl;
47        memwrite = 1'b0;
48        memtoreg = 1'b1;
49        branch = 1'b0;
50        aluop = 2'b00;
51    end
52
53    7'b01100111:
```

```

83 7'b1100011: begin
84      regwrite = 1'b0;
85      alusrc  = 1'b0;
86      memread = 1'b0;
87      memwrite = 1'b0;
88      memtoreg = 1'b0;
89      branch   = 1'b1;
90     aluop    = 2'b01;
91      end
92      7'b11000111: //jalr
93  begin
94      regwrite = 1'b1;
95      alusrc  = 1'b1;
96      memread = 1'b0;
97      memwrite = 1'b0;
98      memtoreg = 1'b0;
99      branch   = 1'b1;
100     aluop    = 2'b10;
101     end
102     7'b11000111: //jal
103  begin
104      regwrite = 1'b1;
105      alusrc  = 1'bX;
106      memread = 1'b0;
107      memwrite = 1'b0;
108      memtoreg = 1'b0;
109      branch   = 1'b1;
110      aluop    = 2'bXX;
111      end
112      endcase
113  end
114 endmodule

```

## ALU\_CONTROL:

```
21 | module alucontrol(
22 |     input logic [1:0] op,
23 |     input logic [2:0] x,
24 |     input logic y,
25 |     output logic [3:0] out
26 | );
27 |     logic [5:0] a;
28 |     always_comb begin
29 |         out = 4'b0000;
30 |         a = {x, op, y};
31 |         case(a)
32 |             6'bxxxx00: out = 4'b0010;
33 |             6'b000100: out = 4'b0010;
34 |             6'b000101: out = 4'b0110;
35 |             6'b111100: out = 4'b0000;
36 |             6'b110100: out = 4'b0001;
37 |             6'b100100: out = 4'b0011;
38 |             6'b010100: out = 4'b0100;
39 |             6'b011100: out = 4'b0101;
40 |             6'b001100: out = 4'b0111;
41 |             6'b101100: out = 4'b1000;
42 |             6'b101101: out = 4'b1001;
43 |             6'b000000: out = 4'b0010;
44 |             6'b111000: out = 4'b0000;
45 |             6'b110000: out = 4'b0001;
46 |             6'b100000: out = 4'b0011;
47 |             6'b010000: out = 4'b0100;
48 |             6'b011000: out = 4'b0101;
49 |             6'b001000: out = 4'b0111;
50 |             6'b101000: out = 4'b1000;
51 |             6'b101001: out = 4'b1001;
52 |         endcase
53 |     end
54 | 
```

## TOP MODULE:

```

20 ///////////////////////////////////////////////////////////////////
21 module Top2#(
22     parameter A = 32,
23     parameter B = 116,
24     parameter E = 32,
25     parameter D = 8,
26     parameter NUM_REG = 32,
27     parameter G = 5,
28     parameter H = 32,
29     parameter I = 7
30 )(
31     input logic clk,
32     input logic reset
33 );
34
35     logic [A-1:0] pc_out , pc_in;
36     logic [E-1:0] instruction;
37     logic we;
38     logic [G-1:0] rsadd1, rsadd2, rdadd;
39     logic [I-1:0] opcode;
40     logic [E-1:0] reg_result1, reg_result2, alu_result , mux_res , imm_res , memresult;
41     logic [2:0] func3;
42     logic [6:0] func7;
43     logic [3:0] op;
44     logic aluSrc;
45     logic memwrite;
46     logic memread;
47     logic memtoreg;
48     logic branch;
49     logic [A-1:0]adderResult;
50     logic [1:0] signal;
51
52     PC pc_inst (
53         .clk(clk),
54         .reset(reset),
55         .pc_in(pc_in),
56         .pc_out(pc_out)
57     );
58
59     InsMemory #(B, E, D) insmemory_inst (
60         .addr(pc_out),
61         .dataR(instruction)
62     );
63
64     Decoder decoder_inst (
65         .instruction(instruction),
66         .opcode(opcode),
67         .rdadd(rdadd),
68         .func3(func3),
69         .rsadd1(rsadd1),
70         .rsadd2(rsadd2),
71         .func7(func7)
72     );
73
74     ALU alu_inst (
75         .op(op),
76         .aluSrc(aluSrc),
77         .rdadd(rdadd),
78         .func3(func3),
79         .rsadd1(rsadd1),
80         .rsadd2(rsadd2),
81         .func7(func7)
82     );
83
84     MUX mux_inst (
85         .mux_res(mux_res),
86         .imm_res(imm_res),
87         .memresult(memresult)
88     );
89
90     Mem mem_inst (
91         .memtoreg(memtoreg),
92         .branch(branch),
93         .adderResult(adderResult)
94     );
95
96     Branch branch_inst (
97         .branch(branch),
98         .adderResult(adderResult)
99     );
100
101 endmodule

```

```

PC pc_inst (
    .clk(clk),
    .reset(reset),
    .pc_in(pc_in),
    .add(pc_out)
);

InsMemory #(B, E, D) insmemory_inst (
    .addr(pc_out),
    .dataR(instruction)
);

Decoder decoder_inst (
    .instruction(instruction),
    .opcode(opcode),
    .rdadd(rdadd),
    .func3(func3),
    .rsadd1(rsadd1),
    .rsadd2(rsadd2),
    .func7(func7)
);

```

```

74      top_control_alu control_unit (
75          .opcode(opcode),
76          .funct3(func3),
77          .funct7(func7[5]),
78          .alu_ctrl(op),
79          .regwrite(we),
80          .alusrc(alusrc),
81          .memread(memread),
82          .memwrite(memwrite),
83          .memtoreg(memtoreg),
84          .branch(branch)
85
86      );
87      Reg_file #(E, NUM_REG, G) regfile_inst (
88          .clk(clk),
89          .we(we),
90          .rs1(rsadd1),
91          .rs2(rsadd2),
92          .rsw(rdadd),
93          .dataw(addrResult),
94          .data1(reg_result1),
95          .data2(reg_result2)
96      );
97      ImmGen imm_gen (
98          .instr(instruction),
99          .imm(imm_res)
100     );
101
102     mux mux_inst (
103         .ri(reg_result2),
104         .li(imm_res),
105         .sl(alusrc),
106         .res(mux_res)
107     );
108
109
110
111
112
113
114
115
116
117

```

```

101
102     mux mux_inst (
103         .ri(reg_result2),
104         .li(imm_res),
105         .sl(alusrc),
106         .res(mux_res)
107     );
108
109     ALU #(H) alu_inst(
110         .A(reg_result1),
111         .B(mux_res),
112         .opcode(op),
113         .result(alu_result)
114     );
115
116     assign signal = {branch, memtoreg};
117

```

```

114      );
115
116      assign signal = {branch , memtoreg} ;
117
118      mux3x1 my_mux(
119          .f(pc_out),
120          .s(memresult),
121          .t(alu_result),
122          .sl(signal),
123          .res(adderResult)
124      );
125      data_mem #(1024) datamem_inst (
126          .clk(clk),
127          .mem_read(memread),
128          .mem_write(memwrite),
129          .funct3(func3),
130          .addr(alu_result),
131          .dataW(reg_result2),
132          .dataR(memresult)
133      );
134
135      Next_PC nextpc0 (
136          .PC_out(pc_out),
137          .imm32(imm_res),
138          .rd1(reg_result1),
139          .opcode(opcode),
140          .next_pc(pc_in)
141      );
142
143 endmodule
144

```





## CONCLUSION:

This lab built a working data path that handles register operations, immediate arithmetic, and memory access. You added load and store support. You decoded S-type and I-type formats. You generated the correct immediate values. You selected between register and immediate operands. You used the ALU to calculate memory addresses. You wrote data to memory on store and read data from memory on load. You routed data back to the register file through a controlled writeback path.

You confirmed correctness with simulation. You observed register updates and memory changes. You verified that control signals match instruction types.

This lab improved your understanding of how instructions translate into hardware operations. You learned how decoding, control logic, memory access, and write-back integrate into one continuous flow in a processor.