## Design Document

Team 1, Section 3
Ethan Bennett, Medhansh Khattar, Harrison Carpenter
Google Docs link

## Design Philosophy

Our design is aimed at reducing the total number of instruction types to implement the relatively prime functionality. By doing so, we plan to limit the number of bits for funct to 1, and opcode to 3 bits. This enables us to allow the use of multiple registers for load-store architecture, and in continuation, branch to farther addresses in 16-bit instructions. Once we decided on 16 bit core instruction formats, we moved onto developing RTLs for each and planning all components required for our datapath in the future.

### Performance

Performance will be measured using execution time of the relPrime function. As such, we have opted to implement a multicycle architecture, as that will decrease the execution time of the program.

## Registers

| Register   | Name   | Description                           |
|------------|--------|---------------------------------------|
| x0         | zero   | Zero constant.                        |
| x1         | ra     | Return address.                       |
| x2         | sp     | Stack pointer.                        |
| x3         | IN     | User input register.                  |
| x4         | OUT    | Output register.                      |
| <b>x</b> 5 | t0     | Temporaries.                          |
| х6         | t1     |                                       |
| <b>x</b> 7 | t2     |                                       |
| x8         | s0, fp | Callee-saved registers.               |
| x9         | s1     |                                       |
| x10        | a0     | Function arguments and return values. |
| x11        | a1     |                                       |
| x12        | a2     |                                       |
| x13        | a3     |                                       |
| x14        | t3     | Another temporary.                    |
| x15        | at     | Assembler temporary.                  |

## Instruction Types

A-Type (Arithmetic)

| 15  | 14  | 13 | 12 | 11  | 10 | 9 | 8 | 7  | 6 | 5 | 4 | 3     | 2   | 1   | 0 |
|-----|-----|----|----|-----|----|---|---|----|---|---|---|-------|-----|-----|---|
| rs2 | rs2 |    |    | rs1 |    |   |   | rd |   |   |   | funct | opc | ode |   |

Allows for two arithmetic operations (addition and subtraction) between registers.

| Instruction | Opcode | Funct | Example           | Meaning         | Description                |  |  |  |
|-------------|--------|-------|-------------------|-----------------|----------------------------|--|--|--|
| add         | 000    | 0     | add x5,<br>x6, x7 | x5 = x6 + x7    | R[rd] = R[rs1]<br>+ R[rs2] |  |  |  |
| sub         | 000    | 1     | sub x5, x6, x7    | x5 = x6 -<br>x7 | R[rd] = R[rs1]<br>- R[rs2] |  |  |  |

I-Type (Immediate)

| 15   | 14       | 13 | 12 | 11  | 10 | 9 | 8 | 7  | 6 | 5 | 4 | 3     | 2   | 1   | 0 |
|------|----------|----|----|-----|----|---|---|----|---|---|---|-------|-----|-----|---|
| Imm[ | Imm[3:0] |    |    | rs1 |    |   |   | rd |   |   |   | funct | opc | ode |   |

Allows for two immediate operations (adding a register value with an immediate, and loading a value to a register from memory).

| Instruction | Opcode | Funct | Example           | Meaning                                                                | Description                  |
|-------------|--------|-------|-------------------|------------------------------------------------------------------------|------------------------------|
| addi        | 001    | 0     | addi x5,<br>x6, 7 | x5 = x6 + 7                                                            | R[rd] = R[rs1] +<br>SE(imm)  |
| lw          | 001    | 1     | lw x5,<br>0(x6)   | Load a value from memory address x6 with an offset of 0 bytes into x5. | R[rd] =<br>M[R[rs1]]+SE(imm) |

S-Type (Store)

| 15 | 14 | 13 | 12 | 11 | 10 | 9 | 8 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|----|----|----|----|----|----|---|---|---|---|---|---|---|---|---|---|
|    |    |    |    |    |    |   |   |   |   | l | 1 |   |   |   |   |

| Imm[8 6:4] rs1 | Imm[3:0 7] | opcode |
|----------------|------------|--------|
|----------------|------------|--------|

Allows storing of a register value to memory.

| Instruction | Opcode | Example      | Meaning                                                          | Description             |
|-------------|--------|--------------|------------------------------------------------------------------|-------------------------|
| sw          | 010    | sw x5, 4(x6) | Store the value from x5 to memory address x6, offset by 4 bytes. | M[R[rs1+SE(imm)]=R[rs2] |

### B-Type (Branch)

| 15  | 14  | 13 | 12  | 11 | 10 | 9 | 8    | 7    | 6 | 5 | 4     | 3   | 2   | 1 | 0 |
|-----|-----|----|-----|----|----|---|------|------|---|---|-------|-----|-----|---|---|
| rs2 | rs2 |    | rs1 |    |    |   | Imm[ | 3:0] |   |   | funct | opc | ode |   |   |

Allows for two types of branches, being branch equal (beq) and branch greater than (bgt). Branch not equal and branch inclusive inequality is implemented by branching over unconditional jumps with their opposites. Note that this immediate is in terms of **bytes**.

| Instructio<br>n | Opcode | Funct | Example           | Meaning                                              | Description                        |
|-----------------|--------|-------|-------------------|------------------------------------------------------|------------------------------------|
| beq             | 011    | 0     | beq x5,<br>x6, -8 | If x5 and x6 are equal, go back 8 instructions.      | if(rs1 == rs2)<br>PC += SE(imm)<<1 |
| bgt             | 011    | 1     | bgt x5,<br>x6, 4  | If x5 is greater than x6, go forward 4 instructions. | if(rs1 > rs2)<br>PC += SE(imm)<<1  |

### JAL-Type (Jump-And-Link)

| 15  | 14    | 13 | 12 | 11 | 10 | 9 | 8 | 7  | 6 | 5 | 4 | 3      | 2  | 1    | 0 |
|-----|-------|----|----|----|----|---|---|----|---|---|---|--------|----|------|---|
| Imm | [7:0] | ]  |    |    |    |   |   | rd |   |   |   | Imm[8] | op | code |   |

Allows for unconditional jumps, with a larger range than a branch instruction. Also updates the return address. Note that this immediate is in terms of **bytes**.

| Instructio n | Opcode | Funct | Example             | Meaning                                  | Description                        |
|--------------|--------|-------|---------------------|------------------------------------------|------------------------------------|
| jal          | 100    | 0     | jal ra,<br>relPrime | Jump to relPrime and set return address. | R[rd] = PC + 2<br>PC += SE(imm)<<1 |

### Memory Map

| 0x0000 -<br>0x00FF | Stack |
|--------------------|-------|
| 0x0100 -<br>0x7FFF | Text  |
| 0x8000 -<br>0xFFFF | Data  |

## Addressing Modes

This processor will use both Register and Immediate addressing modes. Immediate bits are not used in A-Type instructions, as those use register addressing. In I-Types, use the given immediate to either add to a register value, or offset a memory address. In S-Types, they are used to offset memory addresses. They are also rearranged to enable symmetry with other instruction types. In B-Types and JAL-Types, they are used to offset the PC (PC-Relative). This operation is PC relative, since moving the instruction around will still result in the same number of instructions jumped from that position.

### Procedure Calling Conventions

| Caller                                               | Callee                                                          |
|------------------------------------------------------|-----------------------------------------------------------------|
| Preserve return address across calls                 | Use given return address                                        |
| Preserve temporary registers across calls            | Freely use temporary registers                                  |
| Freely use save registers across procedures          | Preserve save registers, and restore before returning to caller |
| Preserve essential argument registers before calling | Use argument registers given                                    |

## Procedure calling convention example

callAddTwo: addi sp, sp, -2 #offset stack by 2 bytes

sw ra, 0(sp) #save ra to stack
jal ra, addTwo #JAL to addTwo.

#Don't bother saving a0, we are done with

#it after this call anyway

lw ra, 0(sp) #restore stack
Addi sp, sp, 2 #restore stack

jal zero, O(ra) #JAL back to caller

addTwo: addi a0, 2 #add a0 and 2

jal zero, O(ra) #JAL back to caller

# Machine Language Translation of relPrime

| Address | Assembly                      | Machine Code        | Comment                                          |
|---------|-------------------------------|---------------------|--------------------------------------------------|
| 0x0000  | relPrime:<br>addi sp, sp, -4  | 1100 0010 0010 0001 | # Allocate stack<br>space for local<br>variables |
| 0x0002  | sw ra, 2(sp)                  | 0010 0001 0000 1010 | # Save return<br>address                         |
| 0x0004  | sw a0, 0(sp)                  | 0000 0001 0101 0010 | # Save argument                                  |
| 0x0006  | addi a1, a1, 2                | 0010 1011 1011 0001 | # m = 2                                          |
| 0x0008  | relPrime_loop: addi a2, a0, 0 | 0000 1100 1010 0001 | # Copy 'n' to<br>a2 for gcd<br>function          |
| 0x000A  | addi a3, a1, 0                | 0000 1101 1011 0001 | # Copy 'm' to a3<br>for gcd function             |
| 0x000C  | jal ra, gcd                   | 0000 0101 0000 1100 | # Call gcd(n, m)                                 |
| 0x000E  | beq a0, zero, +4              | 1010 0000 0010 0011 | <pre># if gcd(n, m) == 1, don't update 'm'</pre> |
| 0x0010  | jal zero, update_m            | 0000 0001 1000 0100 | # update 'm'                                     |
| 0x0012  | addi a1, a1, 1                | 0001 1011 1011 0001 | # m = m + 1                                      |
| 0x0014  | jal zero, relPrime_loop       | 1111 1101 0000 0100 | # Loop back                                      |
| 0x0016  | update_m:<br>addi a0, a1, 0   | 0000 1011 1010 0001 | # Prepare return<br>value                        |
| 0x0018  | lw ra, 2(sp)                  | 0100 0010 0001 1001 | # Restore return<br>address                      |

| 0x001A | lw a0, 0(sp)              | 0000 0010 1010 1001 | # Restore 'n'                                               |
|--------|---------------------------|---------------------|-------------------------------------------------------------|
| 0x001C | addi sp, sp, 4            | 0100 0010 0010 0001 | # Deallocate<br>stack space                                 |
| 0x001E | jal zero, O(ra)           | XXXX XXXX X000 0100 | # Return                                                    |
| 0x0020 | gcd:<br>addi sp, sp, -6   | 1010 0010 0010 0001 | # Allocate stack space for local                            |
| 0x0022 | sw ra, 4(sp)              | 0000 0001 0100 0010 | # Save return<br>address                                    |
| 0x0024 | sw a0, 2(sp)              | 0000 1010 0010 0010 | # Save 'a'                                                  |
| 0x0026 | sw a1, 0(sp)              | 0000 1011 0000 0010 | # Save 'b'                                                  |
| 0x0028 | gcd_loop: beq a0, zero, 5 | 0000 1010 0101 0011 | <pre># if a == 0, jump to the jal 5 instructions down</pre> |
| 0x002A | beq a1, zero, update_a    | 0000 1011 0101 1011 | # if b == 0,<br>update 'a'                                  |
| 0x002C | bgt a1, a0, subtract_b    | 1010 1011 0111 1011 | # if a > b, a = a - b                                       |
| 0x002E | sub a0, a0, a1            | 1011 1010 1010 1000 | <pre># subtract a1 from a0, store in a0</pre>               |
| 0x0030 | jal zero, gcd_loop        | 1111 1101 1000 0100 | # loop                                                      |
| 0x0032 | jal zero, end_gcd         | 0000 0011 1000 0100 | # jump to end_gcd                                           |
| 0x0034 | update_a: addi a0, a1, 0  | 0000 1011 1010 0001 | # a = b                                                     |
| 0x0036 | lw a1, 0(sp)              | 0000 0010 1011 1001 | # Restore 'b'                                               |
| 0x0038 | jal zero, gcd_loop        | 1111 1100 0000 0100 | # loop                                                      |

| 0x003A | subtract_b: sub a1, a1, a0 | 1010 1011 1011 1000 | # else, b = b -    |
|--------|----------------------------|---------------------|--------------------|
| 0x003C | jal zero, gcd_loop         | 1111 1011 0000 0100 | # loop             |
| 0x003E | end_gcd:<br>lw ra, 4(sp)   | 0100 0010 0001 1001 | # restore stack    |
| 0x0040 | lw a0, 2(sp)               | 0010 0010 0001 1001 | # restore stack    |
| 0x0042 | lw a1, 0(sp)               | 0000 0010 0001 1001 | # restore stack    |
| 0x0044 | addi sp, sp, 6             | 0110 0010 0010 0001 | # restore stack    |
| 0x0046 | jal zero, O(ra)            | XXXX XXXX X000 0100 | # return to caller |

## Single-Cycle RTL

### add

newPC = PC + 2
PC = newPC
inst = InstMem[PC]
a = Reg[inst[11:8]]
b = Reg[inst[15:12]]
result = a + b
Reg[inst[7:4]] = result

### addi

newPC = PC + 2
PC = newPC
inst = InstMem[PC]
a = Reg[inst[11:8]]
imm = SE(inst[15:12])
result = a + imm
Reg[inst[7:4]] = result

### lw

newPC = PC + 2
PC = newPC
inst = InstMem[PC]
a = Reg[inst[11:8]]
imm = SE(inst[15:12])
result = a + imm
memOut = Mem[result]
Reg[inst[7:4]] = memOut

#### SW

newPC = PC + 2
PC = newPC
inst = InstMem[PC]
a = Reg[inst[11:8]]
b = Reg[inst[15:12]]
imm = SE(inst[7:4])
result = a + imm
Mem[result] = b

#### beq

newPC = PC + 2
PC = newPC
inst = InstMem[PC]
a = Reg[inst[11:8]]
b = Reg[inst[15:12]]
imm = SE(inst[7:4])<<1
target = PC + imm
if (a == b) PC = target

### bgt

newPC = PC + 2
PC = newPC
inst = Mem[PC]
a = Reg[inst[11:8]]
b = Reg[inst[15:12]]
imm = SE(inst[7:4])<<1
target = PC + imm
if (a > b) PC = target

newPC = PC + 2
PC = newPC
inst = InstMem[PC]
a = Reg[inst[11:8]]
b = Reg[inst[15:12]]
result = a - b

Reg[inst[7:4]] = result

newPC = PC + 2
PC = newPC
inst = Mem[PC]
imm = SE(inst[15:8])<<1
PC = PC + imm

# Generic Component Specifications (Single Cycle)

| Component             | Inputs                                                              | Outputs                  | Behavior                                                                                                                                                                   | RTL<br>Symbols                                                  |
|-----------------------|---------------------------------------------------------------------|--------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------|
| Register              | newPC [15:0]                                                        | inst_address<br>[15:0]   | Takes a 16 bit value on the rising clock edge, and outputs that same value.                                                                                                | Single+ Multicycle: PC Multicycle exclusive: IR,MDR,A,B, ALUOut |
| Register File         | rs1[3:0]<br>rs2[3:0]<br>rd[3:0]<br>reg_data[15:0]<br>reg_write[0:0] | reg_a[3:0]<br>reg_b[3:0] | Takes 3 register addresses and can read rs1 and rs2 into reg_a and reg_b, as well as write a 16 bit value to rd. Also has a 1 bit input that enables writing to registers. | Reg                                                             |
| Instruction<br>Memory | inst_address<br>[15:0]                                              | inst[15:0]               | Takes a 16 bit instruction address and outputs the instruction                                                                                                             | InstMem                                                         |

|                        |                                                          |                              | at that location in memory.                                                                                                        |               |
|------------------------|----------------------------------------------------------|------------------------------|------------------------------------------------------------------------------------------------------------------------------------|---------------|
| Data Memory            | data_address [15:0] data_data* [15:0] data_write [0:0]   | data_out[15:0]               | Takes a 16 bit memory address and outputs the value at it, and can write 16 bits of data to that address if data_write is enabled. | DataMem       |
| Immediate<br>Generator | inst[15:0]                                               | imm[15:0]                    | Reads a 16 bit instruction and constructs a corresponding 9 bit immediate then sign extends it to 16 bits.                         | ImmGen,<br>SE |
| ALU                    | operand_a<br>[15:0]<br>operand_b<br>[15:0]<br>aluop[0:0] | aluout[15:0]<br>aluzero[0:0] | Performs an arithmetic operation between its two 16 bit operands, and outputs the result as a 16 bit signed integer.               | +, -          |
| PCAdder                | PC[15:0]<br>offset[15:0]                                 | newPC[15:0]                  | Adds 2 OR an immediate offset to the program counter to increment to the next instruction.                                         | PCInc         |

<sup>\*</sup>change this later; it sounds stupid

# Generic Component Specifications (Multi Cycle)

| Component              | Inputs                                                              | Outputs                  | Behavior                                                                                                                                                                   | RTL<br>Symbols               |
|------------------------|---------------------------------------------------------------------|--------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------------------|
| Register               | newPC [15:0]                                                        | inst_address<br>[15:0]   | Takes a 16 bit value on the rising clock edge, and outputs that same value.                                                                                                | PC,<br>IR,MDR,A,B,<br>ALUOut |
| Register File          | rs1[3:0]<br>rs2[3:0]<br>rd[3:0]<br>reg_data[15:0]<br>reg_write[0:0] | reg_a[3:0]<br>reg_b[3:0] | Takes 3 register addresses and can read rs1 and rs2 into reg_a and reg_b, as well as write a 16 bit value to rd. Also has a 1 bit input that enables writing to registers. | Reg                          |
| Immediate<br>Generator | inst[15:0]                                                          | imm[15:0]                | Reads a 16 bit instruction and constructs a corresponding 9 bit immediate then sign extends it to 16 bits.                                                                 | ImmGen,<br>SE                |
| ALU                    | operand_a [15:0] operand_b [15:0] aluop[0:0]                        | aluout[15:0]             | Performs an arithmetic operation between its two 16 bit operands, and outputs the result as a 16 bit signed integer.                                                       | +, -                         |
| Memory                 | mem_address[15:0] mem_data[15:0] mem_write[0:0]                     | mem_out[15:0]            | Takes a 16 bit memory address and outputs the data at that location in memory. Can                                                                                         | Mem                          |

|  | also write 16 bits of data to that address if mem_write is enabled. |
|--|---------------------------------------------------------------------|
|--|---------------------------------------------------------------------|

## Multi-Cycle RTL

| А-Туре                        | І-Туре                                                                       | S-Type              | В-Туре               | JAL-Type |  |
|-------------------------------|------------------------------------------------------------------------------|---------------------|----------------------|----------|--|
| IR <= Mem[PC]<br>PC <= PC + 2 |                                                                              |                     |                      |          |  |
| B <= Reg[IR[15:               | A <= Reg[IR[11:8]] B <= Reg[IR[15:12]] ALUOut <= PC + 2 + SE(Immediate*2)<<1 |                     |                      |          |  |
| ALUOut <= A op B              | A ALUOut <= A + SE(Immediate*2)                                              |                     | PC <=<br>ALUZero?A:B |          |  |
| Reg[IR[7:4]]<br><= ALUOut     | MDF <=<br>Mem[ALUOut]                                                        | Mem[ALUOut]<br><= B |                      |          |  |
|                               | Reg[IR[7:4]]<br><= MDR                                                       |                     |                      |          |  |

## Prospective Datapath



## Control Signal Descriptions

| PCSel      | 1 bit  | Controls if the PC is incremented by 4, or by an immediate offset. |
|------------|--------|--------------------------------------------------------------------|
| MemAddr    | 1 bit  | Controls source<br>memory address (PC<br>or ALUOUT)                |
| MemDataSel | 1 bit  | Controls memory data input (MDR or ALUOUT)                         |
| RegDataSel | 1 bit  | Controls register data input (MDR or ALUOUT)                       |
| MemWrite   | 1 bit  | Enables write to memory                                            |
| RegWrite   | 1 bit  | Enables write to register file                                     |
| SRCA       | 1 bit  | Chooses ALU Source A                                               |
| SRCB       | 2 bits | Chooses ALU Source B                                               |
| ALUOP      | 1 bit  | Chooses ALU<br>Operation                                           |
| IREN       | 1 bit  | Enables writing to IR register                                     |

### Control Signal State Diagrams



### Test Checklist:

| Component | Test                                               | Result              | Description                                 |
|-----------|----------------------------------------------------|---------------------|---------------------------------------------|
| ALU       | 5 + 5                                              | 10                  | Positive and<br>Positive                    |
|           | 5 + -4                                             | 1                   | Positive and negative, nonzero              |
|           | -5 + -5                                            | -10                 | Negative and<br>Negative                    |
|           | 5 + -5                                             | 0                   | Positive and<br>Negative, zero              |
|           | 5 - 4                                              | 1                   | Positive and positive, nonzero              |
|           | 5 - 5                                              | 0                   | Positive and<br>Positive, zero              |
|           | 55                                                 | 10                  | Positive and negative                       |
|           | -54                                                | -1                  | Negative and<br>Negative,<br>nonzero        |
|           | -55                                                | 0                   | Negative and<br>Negative, zero              |
| PC        | PC_IN = 1                                          | PC_OUT =            | Test writing to PC.                         |
| Memory    | Memory Address = 299<br>Data = 299<br>MemWrite = 1 | Memory<br>Out = 299 | Write to memory, and read from that address |
|           | Memory Address 299<br>Data = 1<br>MemWrite = 0     | Memory<br>Out = 299 | Read from<br>address 299                    |
|           | Memory Address 299<br>Data = 1<br>MemWrite = 1     | Memory<br>Out = 1   | Overwrite<br>address 299                    |

### Integration Test Plan:

We will test in the following combinations: PC & ALU, PC & MEM & REG, REG & ALU, ALU & MEM. Each set of tests will check both common and edge cases. Specifically, for PC & ALU, we will test incrementing the current value of the PC. For PC & MEM & REG, we will test fetching an instruction from memory and decoding that instruction. For ALU and MEMORY, we will test writing to and reading from a memory address (not from the PC).

# Python Scripts for testing Single RTLs (For Code refer to Appendix.1)

The Python scripts developed for RTL module simulation encompass all operations are processor needs to perform. Here's a closer look at how these scripts function:

### **Environment Initialization:**

- Instruction Memory (InstMem): A list representing the instruction memory, sized to 1024 elements, stores the encoded instructions.
- Registers (Reg): A list of 16 elements represents the general-purpose registers in the system.
- Data Memory (Mem): Similar to InstMem, this 1024-element list serves as the data memory for the system.
- Program Counter (PC): Initialized to 0, PC keeps track of the current instruction being executed.

### Development of Helper Functions:

- Reset Function (reset()): This function is pivotal for reinitializing the simulation environment. It resets the content of registers and memory arrays to zero and sets the Program Counter (PC) back to its initial state. This is crucial for ensuring that each test starts with a clean slate, thereby eliminating any residual effects from previous tests.
- Sign-Extension Function (SE(value)): A critical component in handling immediate values, SE takes an 8-bit value and extends it to 16 bits. This is especially important for branching and immediate arithmetic operations, where the immediate values need to align correctly with the register sizes.

### Core Instruction Functions:

- addi(): Implements an 'add immediate' operation, fetching the instruction from InstMem, extracting operands, and performing the addition.
- add(), sub(): Similar to addi, but for register-to-register addition and subtraction.
- sw(), lw(): Store and load operations, handling data transfer between registers and memory.
- bgt(), beq(): Branch instructions, modifying the PC based on comparison results.
- jal(): A jump instruction, altering the flow of execution by modifying the PC.

### Testing and Validation:

- The scripts include a run\_tests() function that systematically executes each instruction with predefined scenarios, using assertions to validate the outcomes.
- For instance, in testing addi, the script simulates the addition of an immediate value to a register's content and checks if the result is as expected.

### Mini-Program Simulation and Instruction Encoding:

- The mini-program serves as a practical example of how the instruction set can be utilized to perform a real-world task, such as summing an array.
- Instruction Encoding (encode\_instruction(opcode, r1, r2, imm)):
  - This function is fundamental to the mini-program. It encodes the instructions by compactly packing the opcode, register identifiers, and immediate values into a single instruction format.
  - The encoded instructions are then stored in InstMem, ready to be executed sequentially.

### Execution Flow:

- The mini-program is designed to initialize an array in memory and calculate its sum using the lw (load), add, and bgt (branch) instructions.
- It demonstrates the integration of multiple instructions and showcases how the system handles memory operations, arithmetic operations, and conditional branches.
- Debugging and Validation:

- Enhanced print statements are used within the mini-program to provide real-time feedback on the execution flow, which aids in debugging and understanding how each instruction alters the state of the simulation.
- The final assertion checks if the calculated sum matches the expected result, validating the correct execution of the mini-program.

APPENDIX (please put all required stuff above this, atm anything beyond this is the appendix)

#### A.1

```
InstMem = [0] * 1024 # Instruction memory
Reg = [0] * 16
Mem = [0] * 1024 # Data memory
# Program Counter
PC = 0
def reset():
def SE(value):
def addi():
def add():
```

```
def sw():
def lw():
def bgt():
def beq():
```

```
def jal():
def sub():
   a = Reg[(inst >> 8) \& 0xF]
def run tests():
R1=1
```

```
R1=1, R2=2
R1=1, R2=5
```

```
sub()
def encode instruction(opcode, r1, r2, imm):
def mini program test():
```

```
R4 for array size
             print(f"Executing instruction at PC={PC}, inst={inst:04x},
           lw()
```

```
bgt()
    else:
        print(f"Unrecognized opcode: {opcode}")
        break

# Verify the result
    expected_sum = sum(array)
        assert Reg[1] == expected_sum, f"Mini-program failed, expected sum
{expected_sum}, got {Reg[1]}"
        print(f"Mini-program passed, correctly calculated the sum of {array}
as {Reg[1]}")

if __name__ == "__main__":
    run_tests()
    print("Running Mini-Program Test")
    mini_program_test()
```