# Team17 BananaSlug Documentation

## Contents

| 1        | $\mathbf{CP}$ | U Registers           |  |  |  |  |  |
|----------|---------------|-----------------------|--|--|--|--|--|
|          | 1.1           | _                     |  |  |  |  |  |
|          | 1.2           |                       |  |  |  |  |  |
| <b>2</b> | CP            | U Instructions        |  |  |  |  |  |
|          | 2.1           | I-Type Instructions   |  |  |  |  |  |
|          | 2.2           | R-Type Instructions   |  |  |  |  |  |
| 3        | Ado           | dress Space           |  |  |  |  |  |
|          | 3.1           | Permissions           |  |  |  |  |  |
|          | 3.2           | RAM Address Space     |  |  |  |  |  |
|          | 3.3           | IO Address Space      |  |  |  |  |  |
|          | 3.4           | SLUG Address Space    |  |  |  |  |  |
| 4        | Ю             | 9                     |  |  |  |  |  |
|          | 4.1           | Game Input IO         |  |  |  |  |  |
|          | 4.2           | Debug IO              |  |  |  |  |  |
| 5        | Ope           | erating System 10     |  |  |  |  |  |
|          | 5.1           | Reset Sequence        |  |  |  |  |  |
|          | 5.2           | Game Loop Sequence    |  |  |  |  |  |
| 6        | GPU 11        |                       |  |  |  |  |  |
|          | 6.1           | VRAM and Frame Buffer |  |  |  |  |  |
|          | 6.2           | Pixel Encoding        |  |  |  |  |  |
|          | 6.3           | Rendering             |  |  |  |  |  |
| 7        | Dof           | Parangas 16           |  |  |  |  |  |

#### Copyright © 2024 Ethan Sifferman. All rights reserved.

Distribution of this work, in any part, to individuals outside of UCSC Spring 2024 CSE 111 Team17 Members, or UCSC Spring 2024 CSE 111 Instructors, or to those not explicitly granted permission by the copyright holder(s), will result in adherence to the UCSC Academic Misconduct Policy. Legal action may be pursued against individuals to whom the UCSC Academic Misconduct Policy does not apply.

### 1 CPU Registers

#### 1.1 Register File Registers

The Banana CPU has a Register File<sup>1</sup> with 32 addressable 16-bit registers. They are indexed with reg\_a, reg\_b, reg\_c as seen in CPU Instructions. The Zero Register and Stack Pointer Register serve special purposes as defined below.

#### 1.1.1 Zero Register (registers[0])

The Zero Register is a special Register File register at registers [0] that always returns the value zero, and is uneffected by attempted writes.

#### 1.1.2 Stack Pointer Register (registers [29])

The stack is located at memory locations 0x1400-0x33ff, (see Address Space). The Stack Pointer Register is a special Register File register at registers[29] which shall be initialized to 0x3400 on a Reset Sequence.

The Stack Pointer is expected to return to its initial value (0x3400) after a call to either loop() or setup().

The stack works top-down, so when a word is pushed onto the stack, the stack pointer is decreased. When a word is pulled from the stack, the stack pointer is increased. If the stack pointer ever reaches a value outside the stack memory, exection shall stop.

#### 1.2 Program Counter

The Program Counter (PC) is a 16-bit register outside of the Register File which holds the address of the next instruction to be executed. As instructions are executed, the value of the program counter is updated, usually moving on to the next instruction in the sequence. The value can be affected by branch and jump instructions.

The Program Counter is only valid for values greater than 0x8000. When running the Operating System, the Program Counter shall be held at 0x0000. A call to loop() or setup() is done by first setting the Program Counter to 0xfffc, then running a JAL to the desired function address, (see SLUG Address Space). Note that after the successful completion of a call to loop() or setup(), the Program Counter will return to 0x0000.

 $<sup>^1\</sup>mathrm{A}$  Register File is block of memory representing an addressable array of registers.

#### 2 CPU Instructions

All CPU Instructions are 4-bytes long, and have an opcode in their top 6 bits. The opcode must be used to decode the rest of the instruction according to what type of instruction it is: I-Type or R-Type. After each instruction, the Program Counter Register is increased by 4 unless executing a branch or jump instruction.



Figure 1: Instruction Formats

Any unknown instructions shall be treated to mean "No Operation", or NOP. A NOP simply increments the PC by 4, without affecting any other memory or register.

#### 2.1 I-Type Instructions

#### 2.1.1 Or Immediate (ORI)

• **Opcode**: 0

• Operation: R[reg\_b] = R[reg\_a] | Immediate

#### 2.1.2 Add Immediate (ADDI)

• Opcode: 10

• Operation: R[reg\_b] = R[reg\_a] + Immediate

#### 2.1.3 Branch On Equal (BEQ)

• **Opcode**: 12

• Operation: if(R[reg\_a] == R[reg\_b]) PC=PC+4+ 4\*Immediate

#### 2.1.4 Branch On Not Equal (BNE)

• Opcode: 25

• Operation: if(R[reg\_a]!=R[reg\_b]) PC=PC+4+ 4\*Immediate

#### 2.1.5 Store Byte (SB)

• Opcode: 30

• Operation: M[R[reg\_a]+Immediate] = R[reg\_b](7:0)

#### 2.1.6 Load Byte Unsigned (LBU)

- **Opcode**: 40
- Operation: R[reg\_b]=M[R[reg\_a]+Immediate](7:0)

#### 2.1.7 Jump And Link (JAL)

- Opcode: 51
- Operation: R[31]=PC+4; PC=4\*Immediate

#### 2.1.8 Load Word (LW)

- Opcode: 55
- Operation: R[reg\_b] = M[R[reg\_a]+Immediate]

#### 2.1.9 Store Word (SW)

- Opcode: 60
- Operation: M[R[reg\_a]+Immediate] = R[reg\_b]

#### 2.1.10 Jump (J)

- Opcode: 61
- Operation: PC=4\*Immediate

#### 2.2 R-Type Instructions

#### 2.2.1 Nor (NOR)

- Opcode: 28
- Function: 0
- Operation: R[reg\_c] = ~(R[reg\_a] | R[reg\_b])

#### 2.2.2 Set Less Than (SLT)

- Opcode: 28
- Function: 10
- Operation: R[reg\_c] = (R[reg\_a] < R[reg\_b])

#### 2.2.3 Shift Left Logical (SLL)

- Opcode: 28
- Function: 13
- Operation: R[reg\_c] = R[reg\_b] << shift\_value

#### 2.2.4 Shift Right Arithmetic (SRA)

- Opcode: 28
- Function: 16

• Operation: R[reg\_c] = (signed) R[reg\_b] >> shift\_value

#### 2.2.5 Jump Register (JR)

- Opcode: 28 Function: 22
- Operation: PC=R[reg\_a]

#### 2.2.6 Shift Right Logical (SRL)

- Opcode: 28
- Function: 25
- Operation: R[reg\_c] = (unsigned) R[reg\_b] >> shift\_value

#### 2.2.7 Or (OR)

- Opcode: 28
- Function: 31
- Operation: R[reg\_c] = R[reg\_a] | R[reg\_b]

#### 2.2.8 Subtract (SUB)

- Opcode: 28
- Function: 33
- Operation: R[reg\_c] = R[reg\_a] R[reg\_b]

#### 2.2.9 Add (ADD)

- Opcode: 28
- **Function**: 34
- Operation: R[reg\_c] = R[reg\_a] + R[reg\_b]

#### 2.2.10 And (AND)

- Opcode: 28
- Function: 58
- Operation: R[reg\_c] = R[reg\_a] & R[reg\_b]

## 3 Address Space

All data is big-endian.

#### 3.1 Permissions

Permissions are handled as follows:

- "r" indicates a readable address range where load instructions succeed if the load address is readable, and recieve 0 is the load address is not readable.
- "w" denotes a writable address range where store instructions succeed if the store address is writable, and act as NOP if the store address is not writable
- "x" signifies an executable address range where instructions are run if the PC address is executable, and act as NOP if the PC address is not executable.

#### 3.2 RAM Address Space

| Address | Size (bytes) | Permissions | Contents               |
|---------|--------------|-------------|------------------------|
| 0x0000  | 0x7000       | rw          | RAM                    |
| 0x1400  | 0x2000       | rw          | $\operatorname{Stack}$ |
| 0x3400  | 0x3c00       | rw          | VRAM                   |

#### 3.3 IO Address Space

See IO for functions of each address.

| Address | Size (bytes) | Permissions | Contents        |
|---------|--------------|-------------|-----------------|
| 0x7000  | 1            | r           | Controller Data |
| 0x7100  | 1            | r           | Debug stdin     |
| 0x7110  | 1            | W           | Debug stdout    |
| 0x7120  | 1            | W           | Debug stderr    |
| 0x7200  | 1            | W           | Stop Execution  |
|         |              |             |                 |

## $3.4\quad {\rm SLUG~Address~Space}$

| Address | Size (bytes) | Permissions | Contents                   |
|---------|--------------|-------------|----------------------------|
| 0x8000  | 0x8000       | rx          | SLUG File                  |
| 0x8000  | 4            | rx          | "SLUG"                     |
| 0x81e0  | 4            | rx          | Address to setup()         |
| 0x81e4  | 4            | rx          | Address to loop()          |
| 0x81e8  | 4            | rx          | Load Data Address (ROM)    |
| 0x81ec  | 4            | rx          | Program Data Address (RAM) |
| 0x81f0  | 4            | rx          | Data Size                  |

The contents of the SLUG file header are used by the Operating System.

#### 4 IO

#### 4.1 Game Input IO

• Reading 1 byte from 0x7000 shall read the current state of the game controller.

| Bit    | 7 | 6 | 5      | 4     | 3  | 2    | 1    | 0     |
|--------|---|---|--------|-------|----|------|------|-------|
| Button | A | В | SELECT | START | UP | DOWN | LEFT | RIGHT |

```
#define CONTROLLER_A_MASK ((uint8_t)0x80)
#define CONTROLLER_B_MASK ((uint8_t)0x40)
#define CONTROLLER_SELECT_MASK ((uint8_t)0x20)
#define CONTROLLER_START_MASK ((uint8_t)0x10)
#define CONTROLLER_UP_MASK ((uint8_t)0x08)
#define CONTROLLER_DOWN_MASK ((uint8_t)0x04)
#define CONTROLLER_LEFT_MASK ((uint8_t)0x02)
#define CONTROLLER_RIGHT_MASK ((uint8_t)0x01)
```



Figure 2: Banana Controller Diagram

#### 4.2 Debug IO

- Reading from 0x7100 shall request 1 byte from the debug terminal's stdin.
- Writing to 0x7110 shall write 1 byte to the debug terminal's stdout.
- Writing to 0x7120 shall write 1 byte to the debug terminal's stderr.
- $\bullet$  Writing anything to 0x7200 shall stop Banana execution.

## 5 Operating System

The Banana Operating System is responsible for memory management and Slug file execution. Before reading this section, be sure to understand the Banana address space, (See Address Space).

#### 5.1 Reset Sequence

The Reset Sequence shall be run only once, on start-up.

- 1. Clear all of RAM with zeros
- 2. Copy data section to RAM
- 3. Initialize stack pointer register to the end of the stack (0x3400)
- 4. Call setup()
- 5. Begin Game Loop Sequence

#### 5.2 Game Loop Sequence

The Game Loop Sequence shall be run repeatedly, at a target frequency of 60 Hz. (Commonly referred to as 60 frames per second, or FPS). If an iteration of a Game Loop Sequence completes sooner than the target period (~16.667 ms), then there shall be additional delay added before starting the following Game Loop Sequence. If an iteration of a Game Loop Sequence takes longer than the target period (~16.667 ms), then that Game Loop Sequence is completed as normal, then the following Game Loop Sequence begins with no additional delay.

- 1. Run loop()
- 2. The GPU frame buffer is displayed

## 6 GPU

The Graphics Processing Unit (GPU) is responsible for decoding the Video RAM (VRAM) and rendering its contents onto the screen.

#### 6.1 VRAM and Frame Buffer

The GPU only has access to the VRAM address space (0x3400-0x6fff). The VRAM holds a single frame buffer that includes all the data for drawing a single frame.

The display resolution is defined as 128 pixels in width and 120 pixels in height. The pixel address can be calculated as follows:

```
int getPixelAddress(int width, int height) {
    int pixel_index = width + (height * 128);
    int pixel_offset = 1 * pixel_index;
    return 0x3400 + pixel_offset;
}
```

#### 6.2 Pixel Encoding

Each pixel is encoded as a 1-byte integer, denoting the gray level of the pixel. (0x00 translates to Black, and 0xff translates to White).

#### 6.3 Rendering

Upon completion of each loop() iteration by the CPU, the GPU decodes and displays the contents of the frame buffer.

## 7 References

The Banana Controller Diagram used in this work as seen in Figure 2 is derived from a work by Fant0men and is licensed under the Creative Commons Attribution-Share Alike 3.0 Unported license. Modifications have been made to the original work. To view the original work, visit <a href="https://en.wikipedia.org/wiki/File:Nes\_controller.svg">https://en.wikipedia.org/wiki/File:Nes\_controller.svg</a>.