

COLUMBIA UNIVERSITY

EMBEDDED SYSTEMS CSEE 4840

FINAL REPORT

---

# Chip8 Emulator

---

*Authors:*

Ashley KLING  
Levi OLIVER  
Gabrielle TAYLOR  
David WATKINS

*Supervisor:*

Stephen A. EDWARDS

May 12, 2016



# Contents

|          |                                       |          |
|----------|---------------------------------------|----------|
| <b>1</b> | <b>Introduction</b>                   | <b>5</b> |
| <b>2</b> | <b>System Overview</b>                | <b>6</b> |
| 2.1      | Memory Overview . . . . .             | 6        |
| 2.2      | Graphics . . . . .                    | 7        |
| 2.2.1    | Hardware Software Interface . . . . . | 7        |
| 2.3      | Op Codes . . . . .                    | 10       |
| 2.3.1    | 0nnn - SYS addr . . . . .             | 10       |
| 2.3.2    | 00E0 - CLS . . . . .                  | 10       |
| 2.3.3    | 00EE - RET . . . . .                  | 10       |
| 2.3.4    | 1nnn - JP addr . . . . .              | 11       |
| 2.3.5    | 2nnn - CALL addr . . . . .            | 11       |
| 2.3.6    | 3xkk - SE Vx, byte . . . . .          | 11       |
| 2.3.7    | 4xkk - SNE Vx, byte . . . . .         | 11       |
| 2.3.8    | 5xy0 - SE Vx, Vy . . . . .            | 11       |
| 2.3.9    | 6xkk - LD Vx, byte . . . . .          | 11       |
| 2.3.10   | 7xkk - ADD Vx, byte . . . . .         | 11       |
| 2.3.11   | 8xy0 - LD Vx, Vy . . . . .            | 11       |
| 2.3.12   | 8xy1 - OR Vx, Vy . . . . .            | 12       |
| 2.3.13   | 8xy2 - AND Vx, Vy . . . . .           | 12       |
| 2.3.14   | 8xy3 - XOR Vx, Vy . . . . .           | 12       |
| 2.3.15   | 8xy4 - ADD Vx, Vy . . . . .           | 12       |
| 2.3.16   | 8xy5 - SUB Vx, Vy . . . . .           | 12       |
| 2.3.17   | 8xy6 - SHR Vx {, Vy} . . . . .        | 12       |
| 2.3.18   | 8xy7 - SUBN Vx, Vy . . . . .          | 13       |
| 2.3.19   | 8xyE - SHL Vx {, Vy} . . . . .        | 13       |
| 2.3.20   | 9xy0 - SNE Vx, Vy . . . . .           | 13       |
| 2.3.21   | Annn - LD I, addr . . . . .           | 13       |

|          |                                                    |           |
|----------|----------------------------------------------------|-----------|
| 2.3.22   | Bnnn - JP V0, addr . . . . .                       | 13        |
| 2.3.23   | Cxkk - RND Vx, byte . . . . .                      | 13        |
| 2.3.24   | Dbyn - DRW Vx, Vy, nibble . . . . .                | 13        |
| 2.3.25   | Ex9E - SKP Vx . . . . .                            | 14        |
| 2.3.26   | ExA1 - SKNP Vx . . . . .                           | 14        |
| 2.3.27   | Fx07 - LD Vx, DT . . . . .                         | 14        |
| 2.3.28   | Fx0A - LD Vx, K . . . . .                          | 14        |
| 2.3.29   | Fx15 - LD DT, Vx . . . . .                         | 14        |
| 2.3.30   | Fx18 - LD ST, Vx . . . . .                         | 14        |
| 2.3.31   | Fx1E - ADD I, Vx . . . . .                         | 14        |
| 2.3.32   | Fx29 - LD F, Vx . . . . .                          | 14        |
| 2.3.33   | Fx33 - LD B, Vx . . . . .                          | 15        |
| 2.3.34   | Fx55 - LD [I], Vx . . . . .                        | 15        |
| 2.3.35   | Fx65 - LD Vx, [I] . . . . .                        | 15        |
| 2.4      | Keyboard Input . . . . .                           | 15        |
| 2.5      | Sound . . . . .                                    | 16        |
| 2.6      | Screenshots . . . . .                              | 16        |
| <b>3</b> | <b>Hardware Design</b>                             | <b>18</b> |
| 3.1      | CPU, Top and Module Access Control . . . . .       | 18        |
| 3.2      | ALU . . . . .                                      | 20        |
| 3.3      | Random Number Generator . . . . .                  | 20        |
| 3.4      | BCD . . . . .                                      | 20        |
| 3.5      | Graphics and Framebuffer . . . . .                 | 21        |
| 3.6      | Memory and Registers . . . . .                     | 22        |
| 3.7      | Sound . . . . .                                    | 23        |
| 3.8      | Return Address Stack and Program Counter . . . . . | 23        |
| 3.9      | Timers . . . . .                                   | 24        |
| <b>4</b> | <b>Software Overview</b>                           | <b>25</b> |
| 4.1      | Chip8.c . . . . .                                  | 25        |
| 4.2      | Chip8driver.c . . . . .                            | 26        |
| 4.3      | Data Transfer ISA . . . . .                        | 26        |
| <b>5</b> | <b>Project Plan</b>                                | <b>30</b> |
| 5.1      | Lessons Learned . . . . .                          | 30        |
| 5.2      | Timeline . . . . .                                 | 31        |

|          |                                        |           |
|----------|----------------------------------------|-----------|
| <b>6</b> | <b>Debugging</b>                       | <b>32</b> |
| <b>7</b> | <b>Code Listing</b>                    | <b>34</b> |
| 7.1      | SystemVerilog Code . . . . .           | 34        |
| 7.1.1    | bcd.sv . . . . .                       | 34        |
| 7.1.2    | Chip8_CPU.sv . . . . .                 | 35        |
| 7.1.3    | Chip8_rand_num_generator.sv . . . . .  | 60        |
| 7.1.4    | Chip8_Stack.sv . . . . .               | 61        |
| 7.1.5    | Chip8_ALU.sv . . . . .                 | 63        |
| 7.1.6    | Chip8_framebuffer.sv . . . . .         | 65        |
| 7.1.7    | Chip8_VGA_Emulator.sv . . . . .        | 69        |
| 7.1.8    | audio_codec.sv . . . . .               | 75        |
| 7.1.9    | audio_effects.sv . . . . .             | 78        |
| 7.1.10   | Chip8_SoundController.sv . . . . .     | 82        |
| 7.1.11   | i2c_av_config.sv . . . . .             | 84        |
| 7.1.12   | i2c_controller.sv . . . . .            | 87        |
| 7.1.13   | clk_div.sv . . . . .                   | 90        |
| 7.1.14   | timer.sv . . . . .                     | 91        |
| 7.1.15   | Chip8_Top.sv . . . . .                 | 92        |
| 7.1.16   | SoCKit_top.sv . . . . .                | 113       |
| 7.1.17   | utils.svh . . . . .                    | 140       |
| 7.1.18   | enums.svh . . . . .                    | 140       |
| 7.2      | Testbenches . . . . .                  | 143       |
| 7.2.1    | Chip8_CPU_6xkk_7xkk.sv . . . . .       | 143       |
| 7.2.2    | Chip8_CPU_big_testbench.sv . . . . .   | 149       |
| 7.2.3    | Chip8_CPU_testbench.sv . . . . .       | 168       |
| 7.2.4    | Chip8_Top_test.sv . . . . .            | 169       |
| 7.2.5    | delay_timer_testbench.sv . . . . .     | 170       |
| 7.2.6    | Triple_port_reg_file_test.sv . . . . . | 171       |
| 7.2.7    | fb_testbench.sv . . . . .              | 173       |
| 7.2.8    | alu_testbench.sv . . . . .             | 182       |
| 7.3      | Linux Code . . . . .                   | 199       |
| 7.3.1    | chip8.c . . . . .                      | 199       |
| 7.3.2    | chip8driver.c . . . . .                | 212       |
| 7.3.3    | chip8driver.h . . . . .                | 220       |
| 7.3.4    | Makefile . . . . .                     | 225       |
| 7.3.5    | socfpga.dts . . . . .                  | 226       |
| 7.3.6    | usbkeyboard.c . . . . .                | 233       |

|       |                                       |     |
|-------|---------------------------------------|-----|
| 7.3.7 | usbkeyboard.h . . . . .               | 237 |
| 7.4   | Git commit history . . . . .          | 239 |
| 7.5   | Schematics . . . . .                  | 270 |
| 7.5.1 | Chip8_framebuffer.sv . . . . .        | 270 |
| 7.5.2 | clk_div.sv . . . . .                  | 270 |
| 7.5.3 | timer.sv . . . . .                    | 271 |
| 7.5.4 | Chip8_rand_num_generator.sv . . . . . | 272 |
| 7.5.5 | bcd.sv . . . . .                      | 274 |
| 7.5.6 | memory.sv . . . . .                   | 274 |
| 7.5.7 | Chip8_SoundController.sv . . . . .    | 275 |
| 7.5.8 | Entire Design . . . . .               | 276 |

# Chapter 1

## Introduction

Chip-8 is an interpreted programming language from the 1970s. It ran on the COSMAC VIP, and supported many programs such as Pac-Man, Pong, Space Invaders, and Tetris. We aim to create a processor using SystemVerilog and the FPGA on the SoCKit board that runs these programs. During the boot process of the processor chip8 ROM files will be transferred onto the main memory of the processor. The processor will also allow for save states and restoring of states. The processor will handle keyboard inputs and output graphics and sound.



Figure 1.1: A popular Chip8 game: Pong

# Chapter 2

## System Overview

### 2.1 Memory Overview

The Chip-8 specification requires the use of sixteen 8-bit registers (V0-VF), a 16-bit index register, a 64-byte stack with 8-bit stack pointer, an 8-bit delay timer, an 8-bit sound timer, a 64x32 bit frame buffer, and a 16-bit program counter. The Chip8 specification also supported 4096 bytes of addressable memory. All of the supported programs will start at memory location 0x200.

- The sound and delay timers sequentially decrease at a rate of 1 per tick of a 60Hz clock. When the sound timer is above 0, the sound will play as a single monotone beep.
- The framebuffer is an  $(x, y)$  addressable memory array that designates whether a pixel is currently on or off. This will be implemented with a write address, an  $(x, y)$  position, a offset in the x direction, and an 8-bit group of pixels to be drawn to the screen.
- The return address stack stores previous program counters when jumping into a new routine.
- The VF register is frequently used for storing carry values from a subtraction or addition action, and also specifies whether a particular pixel is to be drawn on the screen.



Figure 2.1: Chip8 4K memory layout

## 2.2 Graphics

Important to the specification is the 64x32 pixel display that is associated with the Chip8. Each pixel only contains the information as to whether it is on or off. All setting of pixels of this display are done through the use of sprites that are always  $8 \times N$  where  $N$  is the pixel height of the sprite.

Chip8 comes with a font set (sprites) that allows character 0-9 and A-F to be printed directly to the screen. Each one of these characters fit within a 8x5 grid.<sup>1</sup>

### 2.2.1 Hardware Software Interface

To start running a program, the function `loadROM` and `loadfontset` (located in `chip8.c`) are called. The latter writes the fontset to the Chip8 memory,

---

<sup>1</sup><http://devernay.free.fr/hacks/chip8/C8TECH10.HTM>



Figure 2.2: The screen dimensions for the VGA output

while the former writes a chip8 program (like Pong!) to the chip8 memory.

From the software side, loadfontset copies the pre-defined font set from linux memory to Chip8 memory by writing byte by byte to processor memory, starting from chip8 memory address 0. loadROM copies a chip8 program from the linux memory to the Chip8 memory byte by byte starting at chip8 memory address 0x200, which is the standard initial address for Chip8 programs.

A 51-bit channel exists that writes from the linux side to the hardware. This channel is broken into two parts: an 18-bit channel that is used to tell the chip what type memory to modify or read (stack, memory, registers, etc.), a 32 bit data channel that includes whatever data is needed (writedata, write enable, and sometimes addressing), and a one-bit channel signifying if the call is a read or a write..

From the hardware side, if chipselect goes high, the processor reads data from the bus. It takes in the 18-bit channel (called address), and the 32-bit channel (called writedata). The stage is frozen while input is processed. The address channel tells what needs to be modified. For example, if address==0x19, the processor begins to modify memory. If the write channel is high, it starts to write the data signified by bits writedata[7:0] to memory addressed by writedata[18:9]. (See Chip8\_Top line 348.)

As soon as chipselect goes low again, we resume normal operation of the

| "8"           | Binary | Hex | "9"           | Binary | Hex | "0"           | Binary | Hex | "1"          | Binary   | Hex  |
|---------------|--------|-----|---------------|--------|-----|---------------|--------|-----|--------------|----------|------|
| **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | *            | 00100000 | 0x20 |
| * * 10010000  | 0x90   |     | * * 10010000  | 0x90   |     | * * 10010000  | 0x90   |     | ** 01100000  | 0x60     |      |
| **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | * * 10010000  | 0x90   |     | * 00100000   | 0x20     |      |
| * * 10010000  | 0x90   |     | * 00100000    | 0x10   |     | * * 10010000  | 0x90   |     | * 00100000   | 0x20     |      |
| **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | *** 01110000 | 0x70     |      |

  

| "A"           | Binary | Hex | "B"          | Binary | Hex | "2"           | Binary | Hex | "3"           | Binary | Hex |
|---------------|--------|-----|--------------|--------|-----|---------------|--------|-----|---------------|--------|-----|
| **** 11110000 | 0xF0   |     | *** 11100000 | 0xE0   |     | **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     |
| * * 10010000  | 0x90   |     | * * 10010000 | 0x90   |     | * 00010000    | 0x10   |     | * 00010000    | 0x10   |     |
| **** 11110000 | 0xF0   |     | *** 11100000 | 0xE0   |     | **** 11110000 | 0xF0   |     | *** 11110000  | 0xF0   |     |
| * * 10010000  | 0x90   |     | * * 10010000 | 0x90   |     | * 10000000    | 0x80   |     | * 00010000    | 0x10   |     |
| * * 10010000  | 0x90   |     | *** 11100000 | 0xE0   |     | **** 11110000 | 0xF0   |     | *** 11110000  | 0xF0   |     |

  

| "C"           | Binary | Hex | "D"          | Binary | Hex | "4"           | Binary | Hex | "5"           | Binary | Hex |
|---------------|--------|-----|--------------|--------|-----|---------------|--------|-----|---------------|--------|-----|
| **** 11110000 | 0xF0   |     | *** 11100000 | 0xE0   |     | * * 10010000  | 0x90   |     | **** 11110000 | 0xF0   |     |
| * 10000000    | 0x80   |     | * * 10010000 | 0x90   |     | * * 10010000  | 0x90   |     | * 10000000    | 0x80   |     |
| * 10000000    | 0x80   |     | * * 10010000 | 0x90   |     | **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     |
| * 10000000    | 0x80   |     | * * 10010000 | 0x90   |     | * 00010000    | 0x10   |     | * 00010000    | 0x10   |     |
| **** 11110000 | 0xF0   |     | *** 11100000 | 0xE0   |     | * 00010000    | 0x10   |     | **** 11110000 | 0xF0   |     |

  

| "E"           | Binary | Hex | "F"           | Binary | Hex | "6"           | Binary | Hex | "7"           | Binary | Hex |
|---------------|--------|-----|---------------|--------|-----|---------------|--------|-----|---------------|--------|-----|
| **** 11110000 | 0xF0   |     |
| * 10000000    | 0x80   |     | * 10000000    | 0x80   |     | * 10000000    | 0x80   |     | * 00010000    | 0x10   |     |
| **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | **** 11110000 | 0xF0   |     | * 00100000    | 0x20   |     |
| * 10000000    | 0x80   |     | * 10000000    | 0x80   |     | * * 10010000  | 0x90   |     | * 01000000    | 0x40   |     |
| **** 11110000 | 0xF0   |     | * 10000000    | 0x80   |     | **** 11110000 | 0xF0   |     | * 01000000    | 0x40   |     |

Figure 2.3: Chip8 character sprite specification

processor, restoring all possibly changed values to what they were previously.



Figure 2.4: Overview of the design of the emulator

## 2.3 Op Codes

The Chip8 interpreter works by parsing 16 bit opcodes and operating on the data. All supported op codes in the original Chip8 specification are included.

### 2.3.1 0nnn - SYS addr

Jump to a machine code routine at nnn. This instruction is only used on the old computers on which Chip-8 was originally implemented. It is ignored by modern interpreters. This will not be implemented.

### 2.3.2 00E0 - CLS

Clear the display.

### 2.3.3 00EE - RET

Return from a subroutine. The interpreter sets the program counter to the address at the top of the stack, then subtracts 1 from the stack pointer.

### **2.3.4 1nnn - JP addr**

Jump to location nnn. The interpreter sets the program counter to nnn.

### **2.3.5 2nnn - CALL addr**

Call subroutine at nnn. The interpreter increments the stack pointer, then puts the current PC on the top of the stack. The PC is then set to nnn.

### **2.3.6 3xkk - SE Vx, byte**

Skip next instruction if  $V_x = kk$ . The interpreter compares register  $V_x$  to  $kk$ , and if they are equal, increments the program counter by 2.

### **2.3.7 4xkk - SNE Vx, byte**

Skip next instruction if  $V_x \neq kk$ . The interpreter compares register  $V_x$  to  $kk$ , and if they are not equal, increments the program counter by 2.

### **2.3.8 5xy0 - SE Vx, Vy**

Skip next instruction if  $V_x = V_y$ . The interpreter compares register  $V_x$  to register  $V_y$ , and if they are equal, increments the program counter by 2.

### **2.3.9 6xkk - LD Vx, byte**

Set  $V_x = kk$ . The interpreter puts the value  $kk$  into register  $V_x$ .

### **2.3.10 7xkk - ADD Vx, byte**

Set  $V_x = V_x + kk$ . Adds the value  $kk$  to the value of register  $V_x$ , then stores the result in  $V_x$ .

### **2.3.11 8xy0 - LD Vx, Vy**

Set  $V_x = V_y$ . Stores the value of register  $V_y$  in register  $V_x$ .

### **2.3.12 8xy1 - OR Vx, Vy**

Set  $Vx = Vx \text{ OR } Vy$ . Performs a bitwise OR on the values of  $Vx$  and  $Vy$ , then stores the result in  $Vx$ . A bitwise OR compares the corresponding bits from two values, and if either bit is 1, then the same bit in the result is also 1. Otherwise, it is 0.

### **2.3.13 8xy2 - AND Vx, Vy**

Set  $Vx = Vx \text{ AND } Vy$ . Performs a bitwise AND on the values of  $Vx$  and  $Vy$ , then stores the result in  $Vx$ . A bitwise AND compares the corresponding bits from two values, and if both bits are 1, then the same bit in the result is also 1. Otherwise, it is 0.

### **2.3.14 8xy3 - XOR Vx, Vy**

Set  $Vx = Vx \text{ XOR } Vy$ . Performs a bitwise exclusive OR on the values of  $Vx$  and  $Vy$ , then stores the result in  $Vx$ . An exclusive OR compares the corresponding bits from two values, and if the bits are not both the same, then the corresponding bit in the result is set to 1. Otherwise, it is 0.

### **2.3.15 8xy4 - ADD Vx, Vy**

Set  $Vx = Vx + Vy$ , set  $VF = \text{carry}$ . The values of  $Vx$  and  $Vy$  are added together. If the result is greater than 8 bits (i.e.,  $> 255$ ),  $VF$  is set to 1, otherwise 0. Only the lowest 8 bits of the result are kept, and stored in  $Vx$ .

### **2.3.16 8xy5 - SUB Vx, Vy**

Set  $Vx = Vx - Vy$ , set  $VF = \text{NOT borrow}$ . If  $Vx < Vy$ , then  $VF$  is set to 1, otherwise 0. Then  $Vy$  is subtracted from  $Vx$ , and the results stored in  $Vx$ .

### **2.3.17 8xy6 - SHR Vx {, Vy}**

Set  $Vx = Vx \text{ SHR } 1$ . If the least-significant bit of  $Vx$  is 1, then  $VF$  is set to 1, otherwise 0. Then  $Vx$  is divided by 2.

### **2.3.18 8xy7 - SUBN Vx, Vy**

Set  $Vx = Vy - Vx$ , set  $VF = \text{NOT borrow}$ . If  $Vy < Vx$ , then  $VF$  is set to 1, otherwise 0. Then  $Vx$  is subtracted from  $Vy$ , and the results stored in  $Vx$ .

### **2.3.19 8xyE - SHL Vx {, Vy}**

Set  $Vx = Vx \text{ SHL } 1$ . If the most-significant bit of  $Vx$  is 1, then  $VF$  is set to 1, otherwise to 0. Then  $Vx$  is multiplied by 2.

### **2.3.20 9xy0 - SNE Vx, Vy**

Skip next instruction if  $Vx != Vy$ . The values of  $Vx$  and  $Vy$  are compared, and if they are not equal, the program counter is increased by 2.

### **2.3.21 Annn - LD I, addr**

Set  $I = nnn$ . The value of register  $I$  is set to  $nnn$ .

### **2.3.22 Bnnn - JP V0, addr**

Jump to location  $nnn + V0$ . The program counter is set to  $nnn$  plus the value of  $V0$ .

### **2.3.23 Cxkk - RND Vx, byte**

Set  $Vx = \text{random byte AND kk}$ . The interpreter generates a random number from 0 to 255, which is then ANDed with the value  $kk$ . The results are stored in  $Vx$ . See instruction 8xy2 for more information on AND.

### **2.3.24 Dxyn - DRW Vx, Vy, nibble**

Display n-byte sprite starting at memory location  $I$  at  $(Vx, Vy)$ , set  $VF = \text{collision}$ . The interpreter reads  $n$  bytes from memory, starting at the address stored in  $I$ . These bytes are then displayed as sprites on screen at coordinates  $(Vx, Vy)$ . Sprites are XOR'd onto the existing screen. If this causes any pixels to be erased,  $VF$  is set to 1, otherwise it is set to 0. If the sprite is positioned so part of it is outside the coordinates of the display, it wraps around to the opposite side of the screen.

### **2.3.25 Ex9E - SKP Vx**

Skip next instruction if key with the value of Vx is pressed. Checks the keyboard, and if the key corresponding to the value of Vx is currently in the down position, PC is increased by 2.

### **2.3.26 ExA1 - SKNP Vx**

Skip next instruction if key with the value of Vx is not pressed. Checks the keyboard, and if the key corresponding to the value of Vx is currently in the up position, PC is increased by 2.

### **2.3.27 Fx07 - LD Vx, DT**

Set  $V_x = \text{delay timer value}$ . The value of DT is placed into Vx.

### **2.3.28 Fx0A - LD Vx, K**

Wait for a key press, store the value of the key in Vx. All execution stops until a key is pressed, then the value of that key is stored in Vx.

### **2.3.29 Fx15 - LD DT, Vx**

Set delay timer = Vx. Delay Timer is set equal to the value of Vx.

### **2.3.30 Fx18 - LD ST, Vx**

Set sound timer = Vx. Sound Timer is set equal to the value of Vx.

### **2.3.31 Fx1E - ADD I, Vx**

Set  $I = I + V_x$ . The values of I and Vx are added, and the results are stored in I.

### **2.3.32 Fx29 - LD F, Vx**

Set  $I = \text{location of sprite for digit } V_x$ . The value of I is set to the location for the hexadecimal sprite corresponding to the value of Vx. See section 2.4,

Display, for more information on the Chip-8 hexadecimal font. To obtain this value, multiply VX by 5 (all font data stored in first 80 bytes of memory).

### 2.3.33 Fx33 - LD B, Vx

Store BCD representation of Vx in memory locations I, I+1, and I+2. The interpreter takes the decimal value of Vx, and places the hundreds digit in memory at location in I, the tens digit at location I+1, and the ones digit at location I+2.

### 2.3.34 Fx55 - LD [I], Vx

Stores V0 to VX in memory starting at address I. I is then set to  $I + x + 1$ .

### 2.3.35 Fx65 - LD Vx, [I]

Fills V0 to VX with values from memory starting at address I. I is then set to  $I + x + 1$ .

## 2.4 Keyboard Input

The keyboard input was a 16-key keyboard with keys 0 – 9, A – F. There are a series of op codes (listed in the previous section) that use these key presses. In the design associated with this emulator, the keyboard input will be read in from the ARM processor running Linux and streamed to the emulator.



Figure 2.5: Chip8 16-key keyboard specification

For this project we also added keyboard input for pausing, starting, and resetting the device. The "P" key is mapped to pause, the "O" key is mapped to reset, and the "Enter" key is mapped to run.

## 2.5 Sound

Chip-8 provides 2 timers, a delay timer and a sound timer. The delay timer is active whenever the delay timer register (DT) is non-zero. This timer does nothing more than subtract 1 from the value of DT at a rate of 60Hz. When DT reaches 0, it deactivates.

The sound timer is active whenever the sound timer register (ST) is non-zero. This timer also decrements at a rate of 60Hz, however, as long as ST's value is greater than zero, the Chip-8 buzzer will sound. When ST reaches zero, the sound timer deactivates.

The output of the sound generator has one tone. In the following implementation it will have a soft tone so as to not aggravate the user.<sup>2</sup>

## 2.6 Screenshots



Figure 2.6: Tapeworm game running on the device

---

<sup>2</sup><http://devernay.free.fr/hacks/chip8/C8TECH10.HTM>



Figure 2.7: Chip8 logo ROM running on the device



Figure 2.8: Brick game running on the device

# Chapter 3

## Hardware Design



Figure 3.1: Overview of the hardware design

### 3.1 CPU, Top and Module Access Control

Chip8\_Top.sv handles input from the linux side and its effects on the hardware. The linux side reads from and writes to the hardware on startup and requires access to the register file, as well as the memory. The CPU also

requires access to memory and the register file. To deal with the multiple pieces requiring access to multiple modules, Chip8\_Top.sv handles arbitration, deciding what gets access to what and when. It controls access to the framebuffer, memory, register file, sound controller, delay timers, and the stack. Access to all of these are required by both the C code, as well as the CPU, so all requests are put through Chip8\_Top.

Chip8\_Top is mostly a large arbitration unit, but it also has some functionality. It contains the I-register (see section Memory and Register File), which is important in branching, jumping, and draw-sprite instructions. It also manages the program counter and instruction loading. It is important to note that instructions are 16 bits each, so each time the program counter increments by a single instruction, the value of the program counter increments at 2. Since Chip8 programs start at address 0x200, the program counter starts at this value.

The CPU (Chip8\_CPU.sv) is completely combinational. It has no latches and is contained in a very large always\_comb block. However, it does not execute any instructions in a single 50MHz clock cycle. We cut the effective cycle time of the chip down from 50MHz down to 1kHz, as is appropriate for real Chip8 systems. We made a variable in Chip8\_Top.sv called stage. stage increments to a value of 50,000 before it resets to zero.

When stage is 0 or 1, the processor is loading the next instruction to execute from memory. When stage is greater 1, the processor is executing the instruction. The instruction that is using the most stages is instruction 0x00E0—clear screen, which keeps working until stage is 8189 (which is approximately  $2048 \approx 2$ ). While this large value of stage is not necessary, it uses a minimal amount of logic for this instruction. During most values of stage, nothing is actually happening other than waiting.

As stated previously, the CPU is completely combinational. Many instructions are broken apart into a few steps. For example, 0x7XKK first has a 4-cycle period of waiting for data to come in, and then a one-cycle period of writing the correct output. When an instruction is loaded and being executed, Chip8\_CPU.sv sets all pins to the correct values and Chip8\_Top.sv connects those pins to the appropriate destinations.

For example, say we are in stage 40,000. No instructions run during this period. Say the program counter is equal to 0x2C2. The next instruction is in memory at address 0x2C4, and that instruction is equal to 0x63E1. At stage 50,000, the program counter is set equal to 0x2C4. Then, stage is reset to 0. During stages 0 and 1, the instruction is set equal to the values in memory

at 0x2C4 and 0x2C5. In this case, that instruction is 0x63E1. This is the 0x6XKK instruction, which sets  $V_x = kk$ . See the section on opcodes to read more about what this instruction does. During stages 2 and 3, the CPU will set `reg_addr1` to 0x3, and `reg_writedata1` to 0xE1. It will also set `reg_WE1` high to enable writing. These values will be picked up by `Chip8_Top` and directed to the register file to assign the value 0xE1 to register V3.

`Chip8_CPU` has direct access to the random number generator, the binary-coded-decimal converter, and the ALU, all of which take in and output values combinatorially.

## 3.2 ALU

In order to reduce the number of operations that the CPU used, we implemented an ALU that has functions for addition, subtraction, AND, OR, XOR, left shift by 1, and right shift by 1. The ALU takes in two 16 bit inputs, and results are truncated when needed by the CPU.

## 3.3 Random Number Generator

The random number generator is a 16 bit random number output that starts with an initial value and does a naive xor loop over the 16 bit number on each clock cycle. This is critical for the CXKK instruction which requires a random number anded with the KK byte.

## 3.4 BCD

The binary to BCD conversion was implemented combinatorially using a sequence of bit shifts to extract the ones, tens and hundreds place from an 8 bit binary number. The algorithm works by shifting the binary encoding of the number to the left and then determining as each bit is shifted if the value is too large for a BCD digit (between 0 and 9). This occurs because with each successive left shift, the original binary value is doubled. This is mitigated by checking the value of each BCD digit before shifting. If the value of a BCD digit is greater than 4, then 3 is added to carry the value over into the next digit.

## 3.5 Graphics and Framebuffer

The screen of the original Chip8 system was 64x32 (which is 2048 pixels). It uses sprite-based drawing, but in a slightly strange way—it draws images by XORing the value of what is to be drawn with the value of the existing display. This is interesting and useful—if a sprite is to be erased, one would simply draw the sprite again in the exact location. However, this rapid drawing and erasing leads to rapidly blinking displays.

To actually display the screen, we adapted Professor Edwards's VGA\_LED\_Emulator code. The screen displayed is centered in the VGA monitor, with each bit of framebuffer memory representing an eight-by-eight square of pixels (see figure below). Our adaptation of the VGA\_LED\_Emulator uses simple arithmetic and bit shifting to make sure that each VGA coordinate is properly translated to the correct location in framebuffer memory.



Figure 3.2: Double buffering diagram for the framebuffer

To reduce the amount of blinking, we implemented a double-buffered framebuffer. We used the pre-built Quartus MegaFunction Wizard to create a 2048-bit dual-ported memory file with bit-level granularity, which we used twice. Using the Wizard greatly reduced compile time. The first of these was written to by the CPU whenever the draw command was called. There was only a single opcode that ever wrote to the framebuffer: opcode 0xDXYN (see section on opcodes for more details). The other memory file was connected to the display, which was constantly requesting data to draw. The buffer attached to the VGA Emulator would only have data copied into it 200,000 cycles (4 milliseconds at our 50 MHz clock frequency) after a draw command was given. However, this could give rise to more problems. Oftentimes, programs have looping draw commands. If draw commands happen more than once every 200,000 cycles, data would never be copied over. To combat this, we added an override: if the time since the last draw ever exceeded 500,000 cycles (10ms), we would force the data to copy over. The draw sprite instruction (0xDXYN) specifies the coordinates that the sprites are supposed to be drawn at in (x-coord, y-coord) format. Because this was the only type of addressing the CPU would do, the framebuffer takes in an x and

y coordinate, and writes to the corresponding location in memory. In the actual memory, bits 0 to 63 represent the first row, 64 to 127 represent the second row, 128 to 193 represent the third row, and so on.

The memory file generated by the Wizard does not have combinational reads—all data coming from a read command comes at the rising edge of the next clock cycle. Because of this, when copying data from the CPU-side memory to the screen-side memory, every address requested by the screen-side memory had to be offset by one. We requested data from address 0 to 2047, but every address that data was requested from was always one more than the address being written to. This completely copies the entire memory, including all edges and corners.

Relevant code: `Chip8_VGA_Emulator.sv`, `Chip8_Framebuffer.sv`, `Framebuffer.v`, `enums.svh`

## 3.6 Memory and Registers

The original Chip8 system used 4096 (0xffff) bytes of memory. Addresses 0 to 511 (0x1ff) were reserved for the interpreter. The first 80 bytes of memory were generally reserved for the fontset (one 5-byte character for each hexadecimal character 0-F) for the display. Address 512 (0x200) was the start of most Chip8 programs. From there, all the way up to the end of memory, the program could use as much space as needed.

The register file was composed of 16 1-byte registers. They are called V0 through VF. V0 through VE are used for general purpose computing, while VF is generally reserved for flags. For example, VF is set to 1 if a certain add instruction (opcode 0x8XY4) overflows past a single byte, and 0 if not. VF is also used to tell if the draw instruction (0xDXYN) has erased any bits.

The Chip8 memory file is very large. We decided to use the MegaFunction Wizard to create our memory file. While it is not the most adaptable, it was much faster than using inferred memory. It is dual-ported. While no instructions actually involve reading from or writing to more than one memory address, we still decided to use a dual-ported memory. This left us with a channel open to send data to the linux side. Since instructions are 16 bits, and each entry is 8 bits, this also allowed us to request both bytes of an instruction in a single cycle.

Because many instructions require reading from two registers at the same time, we used a dual-ported memory for the register file. This allowed us to

get both values simultaneously, as well as write back to the register file (for example, instructions like 0x8XY4:  $V_x = V_x + V_y$ ).

One very important fact to note regarding these memory modules is that they do not have combinational reads. When an address is set, the output only reflects this change an entire cycle afterwards.

Chip8 also uses a special 16-bit register called the I-register. This register was simply declared as a logic[15:0] variable in Chip8\_Top.sv.

## 3.7 Sound

Sound was implemented writing an audio codec that implements the Inter-Integrated Circuit (I<sup>2</sup>C) Protocol. The I<sup>2</sup>C controller and I<sup>2</sup>C configuration modules were implemented in order to correctly interface with the audio hardware on the FPGA. An audio codec contained the necessary clocks used to drive the output, in particular a phase locked loop (PLL) generated using the Quartus Megawizard for the Master clock, which had to operate at a frequency of 11.2896 MHz and was not easily or accurately approximated using a clock divider. The samples for the 440 Hz sine wave were stored in memory on the chip. Since the only sound required is a single beep, this was sufficient. The top level module assigned the remaining signals.

## 3.8 Return Address Stack and Program Counter

The program counter is kept as a register (a register—not a reg datatype) in Chip8\_Top.sv. We calculate what the next program counter should be in stage 12. (See NEXT\_PC\_WRITE\_STAGE in enums.svh.) This is a somewhat arbitrary value. The stage needed to be late enough so that the processor would have enough time to calculate what the next instruction should be. Whenever the stage reaches 12, we calculate what the next PC should be. By default, next\_pc is set to pc+2. Since the memory granularity is a single byte, and instructions are two bytes each, pointing the program counter to the next instruction involves incrementing the program counter by 2. However, most programs involve branches, jumps, and subroutine calls. For that, we have an enum declared in enums.svh. Depending on the instruction, the CPU sets a flag, pc\_src, to different values. Chip8\_Top interprets them as follows: –if pc\_src == PC\_SRC\_ALU, next\_pc is set to the value that the

CPU is writing to it –if `pc_src == PRC_SRC_SKIP`, `next_pc` is set to `pc+4`. This effectively skips the next instruction –if `pc_src == PC_SRC_NEXT`, `next_pc` is set to `pc+2`. This is the default, which increments the PC by one instruction.

There is also a special case for setting `next_pc`. The CPU sets flags regarding the return address stack (RAS). This flag, `stk_op`, acts as follows:

- if the CPU sets `stk_op` to `STACK_HOLD`, the PC is unaffected by the RAS
- if the CPU sets `stk_op` to `STACK_PUSH`, `pc+2` is pushed to the top of the stack
- if the CPU sets `stk_op` to `STACK_POP`, `next_pc` is set to the value on top of the RAS in stage 12, and the top of the RAS is popped off. The stack pointer decreases by 1.

The RAS has 16 entries of 16 bits each. The stack does not push every cycle that `stk_op == STACK_PUSH`, nor does it pop every time it equals `STACK_POP`. These operations only happen when `stk_op` changes from `STACK_HOLD` to either of these operations

## 3.9 Timers

There are two timers used in Chip8. These are the delay timer and sound timer. Both timers function the same way, that is, they are set to a particular value and count down at a rate of about 60 Hz per second. In order to implement this, a simple clock divider was implemented as a separate unit, which was then fed into the delay timer. The clock divider simply counts the relevant number of 50 MHz CPU cycles to equal one 60 Hz cycle, then is high for a single 50 MHz cycle and the counter is reset.

# Chapter 4

## Software Overview



Figure 4.1: Overview of the software design

### 4.1 Chip8.c

The chip8.c file is an executable meant to load ROM files and listen for keyboard input. It is capable of resetting the device and contains the entire font set as a c array. It has a series of functions which operate on and manipulate the device. On each key press event it writes either that the key

is currently pressed and the value (mapped using the mapping mentioned earlier) or writes that no key is currently pressed. Upon each reset, the entire ROM file is rewritten to the device and the fontset is sent over as well. Something that is not supported is changing the runtime speed of the device, but this is something that could be trivially added.

## 4.2 Chip8driver.c

This filters and ensures that command sent to the device of the correct format. It is loaded using standard linux kernel module loading standards. It is loaded using sudo insmod chip8driver.ko after running make in the directory. For a list of appropriate opcodes that can be sent to the device, see the next section.

## 4.3 Data Transfer ISA

- **V0\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V1\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V2\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V3\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V4\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V5\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address

- **V6\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V7\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V8\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **V9\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **VA\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **VB\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **VC\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **VD\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **VE\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address
- **VF\_ADDR** - To write data to a particular register, use iowrite with NNNNNNXX Where NNNNNN is ignored XX is the 8 bits to be written  
To read from a register use ioread with the address

- **I\_ADDR** - To write to the I index register NNNNDDDD DDDD is the 16 bits to write Use ioread to read from the I register
- **SOUND\_TIMER\_ADDR** - To write to the sound timer NNNNNNDD Where DD is the number to write to the sound timer Use ioread to read from the sound timer
- **DELAY\_TIMER\_ADDR** - To write to the delay timer NNNNNNDD Where DD is the number to write to the delay timer Use ioread to read from the delay timer
- **STACK\_POINTER\_ADDR** - To write to the stack pointer NNNNNNDD Where DD is the number to write to the stack pointer Only the last six bits are considered  
Use ioread to read from the stack pointer
- **STACK\_ADDR** - To reset the stack, iowrite
- **PROGRAM\_COUNTER\_ADDR** - To write to the program counter 0000DDDD Where DDDD is the number to write to the program counter  
Use ioread to read from the program counter
- **KEY\_PRESS\_ADDR** - To write a keypress to the Chip8 control unit NNNNNNPD Where D is the number corresponding to a keypress 0-F Where P is whether a key is currently pressed or not (0x1, 0x0)
- **STATE\_ADDR** - To change the state of the Chip8 000000DD Where DD is an 8-bit number corresponding to varying states
  - 0x00 - Running
  - 0x02 - Paused

The state is initially set to loading font set Use ioread to read the state of the Chip8

- **MEMORY\_ADDR** - To write to a location in memory 0000\_0000\_0001\_AAAA\_AAAA\_AAA Where DD is the 8-bit data that is to be written Where AAA is the 12-bit address to write the data Where W is a 1-bit value corresponding to a read or a write To read data from memory, use iowrite with

0000\_0000\_0000\_AAAA\_AAAA\_AAAA\_NNNN\_NNNN Where AAA is the 12-bit address to read the data from

- **FRAMEBUFFER\_ADDR** - In order to write data to the instruction 0000\_0000\_0000\_0000\_IIII\_IIII\_IIII\_IIII Where I corresponds to the 16 bits in the instruction The state must currently be in Chip8\_RUN\_INSTRUCTION In order to read data from the framebuffer 0000\_0000\_0000\_0000\_NXXX\_XXXY\_YYYY Where XX is the x position (6 bits) Where YY is the y position (5 bits) Where NN is ignored

# Chapter 5

## Project Plan

### 5.1 Lessons Learned

- **Ashley** - The lecture part of this course is very good for learning the theoretical aspects of embedded systems, but the most informative part is the homeworks and final project, which give you a feeling for how hardware design really works, especially the pitfalls. It is important to always start early and work often on the homeworks and project (especially the project), and to communicate frequently with the group members so that everyone is aware of exactly what each person is working on and when they think it will be done. Furthermore, I learned that it is imperative to think critically about everything you are doing, even the tools you are using. We had several issues with Quartus and Qsys, and even the lab computers. We also had a significant bug caused by assumptions we made about memory access times. We learned to make sure that we really understood the tools we used, and to test EVERYTHING. And last but not least, as David said, when the long nights take their toll (as they will), the difficulty is abated by the camaraderie of a good group.
- **Levi** - So far, we have learned and been taught how to program and how to fix broken code. But programming is sometimes only half of the job of a programmer. The other half is using tools that are supposedly used to make life easier. We encountered many problems using ModelSim, Qsys, and Quartus, and learned to a good degree how to make them all do what we wanted them to. What I've learned is that it is

just as important to learn and understand the tools being used as it is to program. I've also learned that unless a component's functionality is explicitly stated, it should probably be tested—at least basically. Our biggest and longest-lasting problem came from misunderstanding how the timing of the MegaFunction Wizard-generated memory files worked. I'd also like to repeat what David said—after too many late and frustrating nights in the lab, the only thing that kept us (at least partially) sane was each other. Stupid jokes really help.

- **Gabrielle** - Hardware is particularly tricky to work with because of how little of the internal functioning you can see. Although this is something you know at an intellectual level, and something mentioned by anyone with experience in hardware, it's still different from actually encountering this in practice. One of the more difficult challenges was figuring out why what seemed like reasonable code (in software, perhaps) does not translate well into hardware. Indeed, there was a point during the semester where I had to find a basic logic design textbook and look over some more complicated things than what we studied in Fundamentals. Overall it was an interesting class and project, particularly the fact that this was designing an entire interpreter from scratch and reminded me of the process one would go through to design their own computer, their own processor, etc. 10/10 would take again.
- **David** - This course requires a good comprehensive understanding and appreciation for the design and creation of hardware based programs. I have certainly learned some techniques for going into making a project such as this, but certainly it would be advisable to have as much hardware design knowledge going into this project as possible. I also recommend choosing good teammates which help make the project so much easier. People you can joke around with during the most frustrating parts of the project is an important quality in a good team.

## 5.2 Timeline

During the course of the semester we worked diligently to get as much done as we could on time. Unfortunately some of our predicted goals were too ambitious and it required a big crunch at the end to get a fully working emulator.



Figure 5.1: Git history over the semester

# Chapter 6

## Debugging

As stated earlier on in the semester, about a quarter of the time of the entire project was dedicated to debugging. Because this was our first time using these tools and designing in this way, it may have been even more than a quarter for us. Our project was particularly difficult to test since it was all in hardware—it was a black box situation. We worked to incorporate a versatile set of functions and instructions to read from and write to the board, and this helped us greatly—especially in the initial stages. With this, we were able to find out errors like when the PC was not being incremented properly, or when the return address stack was being misused.



Figure 6.1: Our testbench running during the testing process

It was also difficult to test individual errors, since most of them were in SystemVerilog. If we wanted to test a single section, the code would be changed and a recompile would need to happen. This could be time

consuming and problematic.

We got to a stage in the debugging process when we couldn't think of what could possibly be wrong, and our limited output wasn't telling us anything other than something was being stored incorrectly. We stepped through from the ground up, making tracing the executable's assembly code, re-reading Chip8 documentation, and aggressively testing all of our modules and checking our base assumptions. It was only when we got to the MegaFunction Wizard-generated memory module and questioned our base assumptions on the timing of reading that we were able to diagnose and fix the problem.

Below are two pictures. Both of them are the output resulting from running the same executable.



Figure 6.2: An early stage of IBM during the debugging process



Figure 6.3: IBM after several rounds of debugging

# Chapter 7

## Code Listing

### 7.1 SystemVerilog Code

#### 7.1.1 bcd.sv

```
1 module bcd(
2     input logic [7:0] num,
3     output logic [3:0] hundreds,
4     output logic [3:0] tens,
5     output logic [3:0] ones);
6
7     logic [19:0] shift;
8
9     always_comb begin
10         shift[19:8] = 11'd0;
11         shift[7:0] = num;
12         repeat (8) begin
13             if(shift[11:8] >= 3'd5)
14                 shift[11:8] = shift[11:8] + 3'd3;
15             if(shift[15:12] >= 3'd5)
16                 shift[15:12] = shift[15:12] + 3'd3;
17             if(shift[19:16] >= 3'd5)
18                 shift[19:16] = shift[19:16] + 3'd3;
19             // Shift entire register left once
20             shift = shift << 1;
```

```

21      end
22      hundreds = shift[19:16];
23      tens     = shift[15:12];
24      ones     = shift[11:8];
25  end
26
27 endmodule

```

### 7.1.2 Chip8\_CPU.sv

```

1 ****
2 * CHIP8_CPU.sv
3 *
4 * Contains the code for interpreting and running
5 * instructions as per the Chip8
6 * ISA defined at:
7 * http://devernay.free.fr/hacks/chip8/C8TECH10.HTM
8 *
9 * The main idea behind the instructions is that initially
10 * the CPU will request
11 * data from registers and memory at stage 0, and then
12 * operate on the data
13 * returned by the request at stage i, where i > 0. This way
14 * the CPU can handle
15 * instructions that operate over multiple cycles and return
16 * values that are
17 * a function of the number of cycles that have occurred.
18 *
19 * AUTHORS: David Watkins, Levi Oliver
20 * Dependencies:
21 *   - Chip8_CPU/Chip8_ALU.sv
22 *   - Chip8_CPU/Chip8_rand_num_generator.sv
23 *   - Chip8_CPU/bcd.sv
24 *   - enums.svh
25 *   - utils.svh
26 ****

```

```

21
22 'include "../enums.svh"
23 'include "../utils.svh"
24
25 module Chip8_CPU(
26     input logic cpu_clk,
27     input logic[15:0] instruction,
28     input logic[7:0] reg_readdata1, reg_readdata2,
29                     mem_readdata1, mem_readdata2,
30     input logic[15:0] reg_I_readdata,
31     input logic[7:0] delay_timer_readdata,
32
33     input logic key_pressed,
34     input logic[3:0] key_press,
35
36     input logic[11:0] PC_readdata,
37
38     input logic[31:0] stage,
39
40     input logic fb_readdata,
41
42     input Chip8_STATE top_level_state,
43
44     output logic delay_timer_WE, sound_timer_WE,
45     output logic[7:0] delay_timer_writedata,
46     ↳ sound_timer_writedata,
47
48     output PC_SRC pc_src,
49     output logic[11:0] PC_writedata,
50
51     output logic reg_WE1, reg_WE2,
52     output logic[3:0] reg_addr1, reg_addr2,
53     output logic[7:0] reg_writedata1, reg_writedata2,
54
55     output logic mem_WE1, mem_WE2,
56     output logic[11:0] mem_addr1, mem_addr2,
57     output logic[7:0] mem_writedata1, mem_writedata2,
      output logic      mem_request,

```

```

58
59     output logic reg_I_WE,
60     output logic[15:0] reg_I_writedata,
61
62     output logic stk_reset,
63     output STACK_OP stk_op,
64     output logic[15:0] stk_writedata,
65
66     output logic [4:0]    fb_addr_y, //max val = 31
67     output logic [5:0]    fb_addr_x, //max val = 63
68     output logic          fb_writedata, //data to write to
69     →      addresse.
70     output logic          fb_WE, //enable writing to address
71     output logic          fbreset,
72
73     output logic          bit_overwritten, //VF overwritten
74     output logic          isDrawing,           //Draw
75     →      instruction where VF could be overwritten
76
77     output logic halt_for_keypress
78 );
79
80     logic[15:0] alu_in1, alu_in2, alu_out;
81     ALU_f alu_cmd;
82     logic alu_carry;
83
84     wire[15:0] rand_num;
85     logic[7:0] to_bcd;
86     wire[3:0] bcdHundreds, bcdTens, bcdOnes;
87
88     wire[31:0] stage_shifted_by4_minus1 = (stage >> 32'h4) -
89     → 32'h1;
90     logic[31:0] num_rows_written; //used for sprite writing
91
92     logic [31:0] stageminus16;
93
94     Chip8_rand_num_generator rand_num_generator(cpu_clk,
95     →  rand_num);

```

```

92      bcd binary_to_dec(to_bcd, bcdHundreds, bcdTens,
93      ↳ bcd_ones);
94      Chip8_ALU alu(alu_in1, alu_in2, alu_cmd, alu_out,
95      ↳ alu_carry);
96
97      always_comb begin
98          /*DEFAULT WIRE VALUES BEGIN*/
99          delay_timer_WE           = 1'b0;
100         sound_timer_WE          = 1'b0;
101         delay_timer_writedata   = 8'b0;
102         sound_timer_writedata   = 8'b0;
103         pc_src                  = PC_SRC_NEXT;
104         PC_writedata            = 12'b0;
105         reg_WE1                 = 1'b0;
106         reg_WE2                 = 1'b0;
107         reg_addr1                = 4'b0;
108         reg_addr2                = 4'b0;
109         reg_writedata1           = 8'b0;
110         reg_writedata2           = 8'b0;
111         mem_WE1                 = 1'b0;
112         mem_WE2                 = 1'b0;
113         mem_addr1                = 12'h0;
114         mem_addr2                = 12'h0;
115         mem_request              = 1'b0;
116         mem_writedata1           = 8'h0;
117         mem_writedata2           = 8'h0;
118         reg_I_WE                 = 1'b0;
119         reg_I_writedata          = 16'h0;
120         fb_addr_y                = 5'h0;
121         fb_addr_x                = 6'h0;
122         fb_writedata              = 1'b0;
123         fb_WE                   = 1'b0;
124         fbreset                  = 1'b0;
125         num_rows_written          = 4'h0;
126         bit_overwritten           = 1'b0;
127         halt_for_keypress         = 1'b0;
128         alu_in1                  = 16'h0;
129         alu_in2                  = 16'h0;

```

```

128      alu_cmd          = ALU_f_NOP;
129      to_bcd           = 8'h0;
130      stk_op            = STACK_HOLD;
131      stk_reset         = 1'b0;
132      stk_writedata     = 16'b0;
133      isDrawing         = 1'b0;
134      stageminus16      = stage - 32'd16;
135      /*END DEFAULT VALUES*/
136
137
138      /*BEGIN INSTRUCTION DECODE*/
139      if(top_level_state == Chip8_RUNNING && stage != 32'h0)
140          begin
141              casex (instruction)
142                  // 16'h???: begin
143                      //This instruction is only used on the old
144                      //computers on which Chip-8
145                      //was originally implemented. It is ignored by
146                      //modern interpreters.
147                  // end
148
149                  16'h00E0: begin //00E0 - CLS
150                      //Clear the screen
151                      if(stage == 32'h2) begin
152                          fbreset = 1'b1;
153                      end else if (stage > 32'h2 & stage < 32'd8189)
154                          begin
155                              fb_addr_x = stage[7:2];
156                              fb_addr_y = stage[12:8];
157                              fb_WE = 1'b1;
158                              fb_writedata = 1'b0;
159                              //CPU DONE
160                          end
161                      end
162
163                      //memory module fix May 10
164                      16'h00EE: begin //00EE - RET
165                          //Return from a subroutine.

```

```

162      //The interpreter sets the program counter to
163      //the address at the
164      //top of the stack, then subtracts 1 from the
165      //stack pointer.
166      if(stage >= 32'h3 & stage <=
167          //NEXT_PC_WRITE_STAGE) begin //two stages
168          //b/c stack takes two cycles
169          stk_op = STACK_POP;
170          pc_src = PC_SRC_STACK;
171      end else begin
172          //CPU DONE
173      end
174  end
175
176      //memory module fix May 10
177      16'h1xxx: begin //1nnn - JP addr
178          //Jump to location nnn.
179          //The interpreter sets the program counter to
180          //nnn.
181          if(stage >= 32'h3 & stage <=
182              //NEXT_PC_WRITE_STAGE) begin
183              pc_src = PC_SRC_ALU;
184              PC_writedata = instruction[11:0];
185          end else begin
186              //CPU DONE
187          end
188  end
189
190      //memory module fix May 10
191      16'h2xxx: begin //2nnn - CALL addr
192          //Call subroutine at nnn.
193          //The interpreter increments the stack
194          //pointer, then puts the
195          //current PC on the top of the stack. The PC
196          //is then set to nnn.
197          if(stage >= 32'h3 & stage <=
198              //NEXT_PC_WRITE_STAGE) begin
199              stk_op = STACK_PUSH;

```

```

191           stk_writedata = PC_readdata + 12'h2;
192           pc_src = PC_SRC_ALU;
193           PC_writedata = instruction[11:0];
194       end else begin
195           //CPU DONE
196       end
197   end
198
199   //memory module fix May 10
200   16'h3xxx: begin //3xkk - SE Vx, byte
201       //Skip next instruction if Vx = kk.
202       //The interpreter compares register Vx to kk,
203       //and if they are
204       //equal, increments the program counter by
205       //2.
206
207       if(stage >= 32'h3 & stage <=
208           //NEXT_PC_WRITE_STAGE) begin
209           reg_addr1 = instruction[11:8];
210           if(reg_readdata1 == instruction[7:0])
211               pc_src = PC_SRC_SKIP;
212           else pc_src = PC_SRC_NEXT;
213       end
214
215   end
216
217   //memory module fix May 10
218   16'h4xxx: begin //4xkk - SNE Vx, byte
219       //Skip next instruction if Vx != kk.
220       //The interpreter compares register Vx to kk,
221       //and if they are
222       //not equal, increments the program counter
223       //by 2.
224       if(stage >= 32'h3 & stage <=
225           //NEXT_PC_WRITE_STAGE) begin
226           reg_addr1 = instruction[11:8];
227           if(reg_readdata1 != instruction[7:0])
228               pc_src = PC_SRC_SKIP;

```

```

221           else pc_src = PC_SRC_NEXT;
222       end
223   end
224
225   //memory module fix May 10
226   16'h5xx0: begin //5xy0 - SE Vx, Vy
227       //Skip next instruction if Vx = Vy.
228       //The interpreter compares register Vx to
229       //register Vy, and if
230       //they are equal, increments the program
231       //counter by 2.
232       if(stage >= 32'h3 & stage <=
233           NEXT_PC_WRITE_STAGE) begin
234           reg_addr1 = instruction[11:8];
235           reg_addr2 = instruction[ 7:4];
236           if(reg_readdata1 == reg_readdata2) pc_src
237               = PC_SRC_SKIP;
238           else pc_src = PC_SRC_NEXT;
239       end
240   end
241
242   //memory module fix May 10
243   16'h6xxx: begin //6xkk - LD Vx, byte
244       //Set Vx = kk.
245       //The interpreter puts the value kk into
246       //register Vx.
247
248       if(stage == 32'h2 || stage == 32'h3) begin
249           reg_addr1 = instruction[11:8];
250           reg_writedata1 = instruction[7:0];
251           reg_WE1 = 1'b1;
252       end else begin
253           //CPU DONE
254       end
255   end
256
257   //memory module fix May 10
258   16'h7xxx: begin //7xkk - ADD Vx, byte

```

```

254      //Set Vx = Vx + kk.
255      //Adds the value kk to the value of register
256      ↳ Vx, then stores the
257      //result in Vx.
258      if(stage >= 32'h2 & stage <= 32'h6) begin
259          reg_addr1 = instruction[11:8];
260      end else if(stage == 32'h7) begin
261          reg_addr1 = instruction[11:8];
262          reg_writedata1 = alu_out[7:0];
263          reg_WE1 = 1'b1;
264
265          alu_in1 = reg_readdata1;
266          alu_in2 = instruction[7:0];
267          alu_cmd = ALU_f_ADD;
268      end else begin
269          //CPU DONE
270      end
271
272      //memory module fix May 10
273      //Arithmetic operators
274      16'h8xxx: begin //8xyk
275          if(stage >= 32'h2 && stage <= 32'h7) begin
276              reg_addr1 = instruction[11:8];
277              reg_addr2 = instruction[ 7:4];
278          end else if(stage >= 32'h2 && stage <= 32'h8)
279              begin
280                  case (instruction[3:0])
281                      4'h0: begin //8xy0 - LD Vx, Vy
282                          //Set Vx = Vy.
283                          //Stores the value of register Vy
284                          ↳ in register Vx.
285                          reg_addr1 = instruction[11:8];
286                          reg_addr2 = instruction[ 7:4];
287                          reg_writedata1 = reg_readdata2;
288                          reg_WE1 = 1'b1;
289          end

```

```

289      4'h1: begin //8xy1 - OR Vx, Vy
290          //Set Vx = Vx OR Vy.
291          //Performs a bitwise OR on the
292          //→ values of Vx and Vy,
293          //then stores the result in Vx. A
294          //→ bitwise OR
295          //compares the corrseponding bits
296          //→ from two values,
297          //and if either bit is 1, then
298          //→ the same bit in the
299          //result is also 1. Otherwise, it
300          //→ is 0.
301
302          alu_cmd = ALU_f_OR;
303          alu_in1 = reg_readdata1;
304          alu_in2 = reg_readdata2;
305
306          reg_addr1 = instruction[11:8];
307          reg_addr2 = instruction[ 7:4];
308          reg_WE1 = 1'b1;
309          reg_writedata1 = alu_out[7:0];
310      end
311
312      4'h2: begin //8xy2 - AND Vx, Vy
313          //Set Vx = Vx AND Vy.
314          //Performs a bitwise AND on the
315          //→ values of Vx and Vy,
316          //then stores the result in Vx. A
317          //→ bitwise AND
318          //compares the corrseponding bits
319          //→ from two values,
320          //and if both bits are 1, then
321          //→ the same bit in the
322          //result is also 1. Otherwise, it
323          //→ is 0.
324
325          alu_cmd = ALU_f_AND;
326          alu_in1 = reg_readdata1;

```

```

317     alu_in2 = reg_readdata2;
318
319     reg_addr1 = instruction[11:8];
320     reg_addr2 = instruction[ 7:4];
321     reg_WE1 = 1'b1;
322     reg_writedata1 = alu_out[7:0];
323   end
324
325   4'h3: begin //8xy3 - XOR Vx, Vy
326     //Set Vx = Vx XOR Vy.
327     //Performs a bitwise exclusive OR
328     //→ on the values of
329     //Vx and Vy, then stores the
330     //→ result in Vx. An
331     //exclusive OR compares the
332     //→ corrsponding bits from
333     //two values, and if the bits are
334     //→ not both the same,
335     //then the corresponding bit in
336     //→ the result is set to
337     //1. Otherwise, it is 0.
338
339     alu_cmd = ALU_f_XOR;
340     alu_in1 = reg_readdata1;
341     alu_in2 = reg_readdata2;
342
343     reg_addr1 = instruction[11:8];
344     reg_addr2 = instruction[ 7:4];
345     reg_WE1 = 1'b1;
346     reg_writedata1 = alu_out[7:0];
347   end
348
349   4'h4: begin //8xy4 - ADD Vx, Vy
350     //Set Vx = Vx + Vy, set VF =
351     //→ carry.
352     //The values of Vx and Vy are
353     //→ added together. If the

```

```

347          //result is greater than 8 bits
348          → (i.e., > 255,) VF is
349          //set to 1, otherwise 0. Only the
350          → lowest 8 bits of
351          //the result are kept, and stored
352          → in Vx.

353      alu_cmd = ALU_f_ADD;
354      alu_in1 = reg_readdata1;
355      alu_in2 = reg_readdata2;

356      reg_addr1 = instruction[11:8];
357      reg_WE1 = 1'b1;
358      reg_writedata1 = alu_out[7:0];

359      reg_addr2 = 4'hF;
360      reg_WE2 = 1'b1;
361      reg_writedata2 = alu_carry;
362  end

363
364  4'h5: begin //8xy5 - SUB Vx, Vy
365          //Set Vx = Vx - Vy, set VF = NOT
366          → borrow.
367          //If Vx > Vy, then VF is set to
368          → 1, otherwise 0. Then
369          //Vy is subtracted from Vx, and
370          → the results stored
371          //in Vx.

372      alu_cmd = ALU_f_MINUS;
373      alu_in1 = reg_readdata1;
374      alu_in2 = reg_readdata2;

375      reg_addr1 = instruction[11:8];
376      reg_WE1 = 1'b1;
377      reg_writedata1 = alu_out[7:0];

378      reg_addr2 = 4'hF;

```

```

379           reg_WE2 = 1'b1;
380           reg_writedata2 = alu_carry;
381       end
382
383   4'h6: begin //8xy6 - SHR Vx {, Vy}
384     //Set Vx = Vx SHR 1.
385     //If the least-significant bit of
386     //→ Vx is 1, then VF
387     //is set to 1, otherwise 0. Then
388     //→ Vx is divided by 2.
389
390     reg_addr1 = instruction[11:8];
391     reg_WE2 = 1'b1;
392     reg_writedata2 = {7'h0,
393                       → reg_readdata1[0]};
394
395     alu_cmd = ALU_f_RSHIFT;
396     alu_in1 = reg_readdata1;
397     alu_in2 = 1;
398
399     reg_addr2 = 4'hF;
400     reg_WE1 = 1'b1;
401     reg_writedata1 = alu_out[7:0];
402   end
403
404   4'h7: begin //8xy7 - SUBN Vx, Vy
405     //Set Vx = Vy - Vx, set VF = NOT
406     //→ borrow.
407     //If Vy > Vx, then VF is set to
408     //→ 1, otherwise 0. Then
409     //Vx is subtracted from Vy, and
410     //→ the results stored
411     //in Vx.
412
413     alu_cmd = ALU_f_MINUS;
414     alu_in1 = reg_readdata2;
415     alu_in2 = reg_readdata1;

```

```

411           reg_addr1 = instruction[11:8];
412           reg_WE1 = 1'b1;
413           reg_writedata1 = alu_out[7:0];
414
415           reg_addr2 = 4'hF;
416           reg_WE2 = 1'b1;
417           reg_writedata2 = alu_carry;
418       end
419
420   4'hE: begin //8xyE - SHL Vx {, Vy}
421     //Set Vx = Vx SHL 1.
422     //If the most-significant bit of
423     //→ Vx is 1, then VF is
424     //set to 1, otherwise to 0. Then
425     //→ Vx is multiplied
426     //by 2.
427
428     reg_addr2 = 4'hF;
429     reg_WE2 = 1'b1;
430     reg_writedata2 = {7'h0,
431     → reg_readdata1[7]};
432
433     alu_cmd = ALU_f_LSHIFT;
434     alu_in1 = reg_readdata1;
435     alu_in2 = 1;
436
437     reg_addr1 = instruction[11:8];
438     reg_WE1 = 1'b1;
439     reg_writedata1 = alu_out[7:0];
440   end
441
442   default : /* default */;
443   endcase
444 end else begin
445   //CPU DONE
446 end

```

```

446 //memory module fix May 10
447 16'h9xx0: begin //9xy0 - SNE Vx, Vy
448     //Skip next instruction if Vx != Vy.
449     //The values of Vx and Vy are compared, and
450     //if they are not
451     //equal, the program counter is increased by
452     // 2.
453
454     if(stage >= 32'h3 & stage <=
455         NEXT_PC_WRITE_STAGE) begin
456         reg_addr1 = instruction[11:8];
457         reg_addr2 = instruction[ 7:4];
458         if(reg_readdata1 != reg_readdata2) pc_src
459             = PC_SRC_SKIP;
460         else pc_src = PC_SRC_NEXT;
461     end
462 end
463
464 //memory module fix May 10
465 16'hAxxx: begin //Annn - LD I, addr
466     //Set I = nnn.
467     //The value of register I is set to nnn.
468     if(stage == 32'h2 || stage == 32'h3) begin
469         reg_I_WE = 1'b1;
470         reg_I_writedata = {4'h0,
471                           instruction[11:0]};
472     end else begin
473         //CPU DONE
474     end
475
476 end
477
478 //memory module fix May 10
479 16'hBxxx: begin //Bnnn - JP V0, addr
480     //Jump to location nnn + V0.
481     //The program counter is set to nnn plus the
482     //  value of V0.

```

```

477      if(stage >= 32'h2 & stage <=
478          → NEXT_PC_WRITE_STAGE) begin
479          reg_addr1 = 4'h0;
480          PC_writedata = instruction[11:0] + {4'h0,
481          → reg_readdata1};
482          pc_src = PC_SRC_ALU;
483      end
484
485 //memory module fix May 10
486 16'hCxxx: begin //Cxkk - RND Vx, byte
487 //Set Vx = random byte AND kk.
488 //The interpreter generates a random number
489 //from 0 to 255, which
490 //is then ANDed with the value kk. The
491 //results are stored in Vx.
492 //See instruction 8xy2 for more information
493 //on AND.
494
495 if(stage >= 32'h3 & stage <=
496     → NEXT_PC_WRITE_STAGE) begin
497     alu_cmd = ALU_f_AND;
498     alu_in1 = rand_num[7:0];
499     alu_in2 = instruction[7:0];
500
501     reg_addr1 = instruction[11:8];
502     reg_WE1 = 1'b1;
503     reg_writedata1 = alu_out[7:0];
504 end else begin
505     //CPU DONE
506 end
507
508
509 //memory module fix May 10
510 16'hDxxx: begin //Dxyn - DRW Vx, Vy, nibble
511 //Display n-byte sprite starting at memory
512 //location I at

```

```

508 // (Vx, Vy), set VF = collision.
509
510 // The interpreter reads n bytes from memory,
511 // starting at the
512 // address stored in I. These bytes are then
513 // displayed as sprites
514 // on screen at coordinates (Vx, Vy). Sprites
515 // are XORed onto the
516 // existing screen. If this causes any pixels
517 // to be erased, VF is
518 // set to 1, otherwise it is set to 0. If the
519 // sprite is
520 // positioned so part of it is outside the
521 // coordinates of the
522 // display, it wraps around to the opposite
523 // side of the screen.
524 // See instruction 8xy3 for more information
525 // on XOR, and section
526 // 2.4, Display, for more information on the
527 // Chip-8 screen and
528 // sprites.
529
530 if(stage > 4'b1111) begin
531     reg_addr1 = instruction[11:8];
532     reg_addr2 = instruction[ 7:4];
533     num_rows_written =
534         {7'b0, stageminus16[31:7]};
535     mem_addr1 = num_rows_written +
536         reg_I_readdata;
537     mem_request = 1'b1;
538     fb_addr_x = reg_readdata1 + ({5'b0,
539         stageminus16[6:4]});
540     fb_addr_y = reg_readdata2 + ({4'b0,
541         num_rows_written[3:0]});

542     fb_writedata = mem_readdata1[3'h7 -
543         stageminus16[6:4]] ^ fb_readdata;

```

```

532         fb_WE = (num_rows_written < {28'h0,
533             ↳ instruction[3:0]}) & (&(stage[3:0]));
534         bit_overwritten = (mem_readdata1[3'h7 -
535             ↳ stageminus16[6:4]]) & (fb_readdata) &
536             ↳ fb_WE;
537         isDrawing = 1'b1;
538     end
539     /*
540     if(stage >= 32'h2) begin
541         reg_addr1 = instruction[11:8];
542         reg_addr2 = instruction[ 7:4];
543
544         if(stage <= 32'h15) num_rows_written =
545             ↳ 4'b0;
546         else num_rows_written =
547             ↳ stage_shifted_by4_minus1[3:0];//((stage >> 32'h4) -
548             ↳ 32'h1);
549
550         mem_addr1 = reg_I_readdata[11:0] +
551             ↳ {8'b0,num_rows_written};
552         mem_request = (stage >= 32'd16) &
553             ↳ (stage_shifted_by4_minus1 < instruction[3:0]) &
554                 !(stage[0]);
555         fb_WE = (stage >= 32'd16) &
556             ↳ (stage_shifted_by4_minus1 < instruction[3:0]) &
557                 (stage[0]);
558         fb_addr_x = reg_readdata1 + ({5'b0,
559             ↳ stage[3:1]});
560         fb_addr_y = reg_readdata2 + ({4'b0,
561             ↳ num_rows_written});
562         fb_writedata = mem_readdata1[stage[3:1]]
563             ↳ ^ fb_readdata;
564         bit_overwritten =
565             ↳ (mem_readdata1[stage[3:1]]) & (fb_readdata) & fb_WE;
566             //bit_overwritten goes high whenever
567             ↳ a pixel is set from 1 to 0
568                 isDrawing = 1'b1;
569     end

```

```

554           */
555       end
556
557 //memory module fix May 10
558 16'hEx9E: begin //Ex9E - SKP Vx
559     //Skip next instruction if key with the value
560     //of Vx is pressed.
561     //Checks the keyboard, and if the key
562     //corresponding to the value
563     //of Vx is currently in the down position, PC
564     //is increased by 2.
565
566     if(stage >= 32'h2 & stage <=
567         NEXT_PC_WRITE_STAGE) begin
568         reg_addr1 = instruction[11:8];
569         if(key_pressed && key_press ==
570             reg_readdata1) begin
571             pc_src = PC_SRC_SKIP;
572         end
573     end else begin
574         //CPU DONE
575     end
576   end
577
578 //memory module fix May 10
579 16'hExA1: begin //ExA1 - SKNP Vx
580     //Skip next instruction if key with the value
581     //of Vx is not
582     //pressed.
583     //Checks the keyboard, and if the key
584     //corresponding to the value
585     //of Vx is currently in the up position, PC
586     //is increased by 2.
587
588     if(stage >= 32'h2 & stage <=
589         NEXT_PC_WRITE_STAGE) begin
590         reg_addr1 = instruction[11:8];

```

```

582         if(~key_pressed || key_press !=  

583             → reg_readdata1) begin  

584             pc_src = PC_SRC_SKIP;  

585         end  

586     end else begin  

587         //CPU DONE  

588     end  

589 end  

590  

591 //memory module fix May 10  

592 //F Instructions  

593 16'hFx07: begin //Fx07 - LD Vx, DT  

594     //Set Vx = delay timer value.  

595     //The value of DT is placed into Vx.  

596  

597     if(stage >= 32'h2 & stage <= 32'h6) begin  

598         reg_addr1 = instruction[11:8];  

599         reg_writedata1 = delay_timer_readdata;  

600         reg_WE1 = 1'b1;  

601     end else begin  

602         //CPU DONE  

603     end  

604 end  

605  

606 16'hFx0A: begin //Fx0A - LD Vx, K  

607     //Wait for a key press, store the value of  

608     → the key in Vx.  

609     //All execution stops until a key is pressed,  

610     → then the value of  

611     //that key is stored in Vx.  

612  

613     if((stage >= 32'h2) && (NEXT_PC_WRITE_STAGE >=  

614         → stage) & !halt_for_keypress) begin  

615         halt_for_keypress = 1'b1;  

616     end else if(key_pressed) begin  

617         halt_for_keypress = 1'b0;  

618         reg_addr1 = instruction[11:8];  

619         reg_writedata1 = key_press;

```

```

616           reg_WE1 = 1'b1;
617       end else begin
618           //CPU DONE
619       end
620   end
621
622   //memory module fix May 10
623   16'hFx15: begin //Fx15 - LD DT, Vx
624       //Set delay timer = Vx.
625       //DT is set equal to the value of Vx.
626
627   if(stage >= 32'h2 & stage <= 32'h6) begin
628       reg_addr1 = instruction[11:8];
629   end else if(stage <= NEXT_PC_WRITE_STAGE)
630       begin
631           delay_timer_writedata = reg_readdata1;
632           delay_timer_WE = 1'b1;
633       end else begin
634           //CPU DONE
635       end
636
637   //memory module fix May 10
638   16'hFx18: begin //Fx18 - LD ST, Vx
639       //Set sound timer = Vx.
640       //ST is set equal to the value of Vx.
641
642   if(stage >= 32'h2 & stage <= 32'h6) begin
643       reg_addr1 = instruction[11:8];
644   end else if(stage == 32'h7) begin
645       sound_timer_writedata = reg_readdata1;
646       sound_timer_WE = 1'b1;
647   end else begin
648       //CPU DONE
649   end
650
651
652   //memory module fix May 10

```

```

653 16'hFx1E: begin //Fx1E - ADD I, Vx
654     //Set I = I + Vx.
655     //The values of I and Vx are added, and the
656     //→ results are stored
657     //in I.
658
659     if(stage >= 32'h2 & stage <= 32'h5) begin
660         reg_addr1 = instruction[11:8];
661     end else if(stage == 32'h6) begin
662         alu_cmd = ALU_f_ADD;
663         alu_in1 = reg_I_readdata;
664         alu_in2 = reg_readdata1;
665
666         reg_I_writedata = alu_out;
667         reg_I_WE = 1'b1;
668     end else begin
669         //CPU DONE
670     end
671
672     //memory module fix May 10
673     16'hFx29: begin //Fx29 - LD F, Vx
674         //Set I = location of sprite for digit Vx.
675         //The value of I is set to the location for
676         //→ the hexadecimal
677         //sprite corresponding to the value of Vx.
678         //→ See section 2.4,
679         //Display, for more information on the Chip-8
680         //→ hexadecimal font.
681
682         //The chip8 fontset has each character
683         //→ starting from 0 to 80,
684         //where each character takes 5 bytes each.

685
686     if(stage >= 32'h2 & stage <= 32'h5) begin
687         reg_addr1 = instruction[11:8];
688     end else if(stage == 32'h6) begin

```

```

685           reg_I_writedata = {10'h0,
686                         ↳ reg_readdata1[3:0], 2'h0} +
687                         ↳ reg_readdata1[3:0];
688           reg_I_WE = 1'b1;
689       end else begin
690           //CPU DONE
691       end
692
693   //memory module fix May 10
694   16'hFx33: begin //Fx33 - LD B, Vx
695       //Store BCD representation of Vx in memory
696       ↳ locations I, I+1, and
697       //I+2.
698       //The interpreter takes the decimal value of
699       ↳ Vx, and places the
700       //hundreds digit in memory at location in I,
701       ↳ the tens digit at
702       //location I+1, and the ones digit at
703       ↳ location I+2.
704
705
706   if(stage >= 32'h2 & stage <= 32'h5) begin
707       reg_addr1 = instruction[11:8];
708   end else if(stage >= 32'h6 & stage <= 32'h9)
709       ↳ begin
710           to_bcd = reg_readdata1;
711           reg_addr1 = instruction[11:8];
712
713           mem_addr1 = reg_I_readdata[11:0];
714           mem_request = 1'b1;
715           mem_writedata1 = bcdHundreds;
716           mem_WE1 = 1'b1;
717   end else if(stage >= 32'hA & stage <= 32'hD)
718       ↳ begin
719           to_bcd = reg_readdata1;
720           reg_addr1 = instruction[11:8];
721
722           mem_addr1 = reg_I_readdata + 12'h1;

```

```

715         mem_request = 1'b1;
716         mem_writedata1 = bcd_tens;
717         mem_WE1 = 1'b1;
718     end else if(stage >= 32'hE & stage <= 32'hF1)
719     begin
720         to_bcd = reg_readdata1;
721         reg_addr1 = instruction[11:8];
722
723         mem_addr1 = reg_I_readdata + 12'h2;
724         mem_request = 1'b1;
725         mem_writedata1 = bcd_ones;
726         mem_WE1 = 1'b1;
727     end else begin
728         //CPU DONE
729     end
730
731 //memory module fix May 10
732 16'hFx55: begin //Fx55 - LD [I], Vx
733     //Store registers V0 through Vx in memory
734     // starting at location I
735     //The interpreter copies the values of
736     // registers V0 through Vx
737     //into memory, starting at the address in I.
738     if(stage >= 32'h7 & (stage[18:3] <=
739     // instruction[11:8])) begin
740         reg_addr1 = stage[6:3];
741
742         alu_cmd = ALU_f_ADD;
743         alu_in1 = reg_I_readdata;
744         alu_in2 = stage[18:3];
745
746         mem_addr1 = alu_out[11:0];
747         mem_request = 1'b1;
748         mem_writedata1 = reg_readdata1;
749         mem_WE1 = &(stage[2:0]);
750
751     end

```

```

749
750     end
751
752     //memory module fix May 10
753     16'hFx65: begin //Fx65 - LD Vx, [I]
754         //Read registers V0 through Vx from memory
755         //→ starting at location
756         //I.
757         //The interpreter reads values from memory
758         //→ starting at location
759         //I into registers V0 through Vx.
760         if(stage >= 32'h7 & (stage[18:3] <=
761             //→ instruction[11:8])) begin
762             reg_addr1 = stage[6:3];
763
764             alu_cmd = ALU_f_ADD;
765             alu_in1 = reg_I_readdata;
766             alu_in2 = stage[18:3];
767
768             mem_addr1 = alu_out[11:0];
769             mem_request = 1'b1;
770
771             reg_writedata1 = mem_readdata1[7:0];
772             reg_WE1 = &(stage[2:0]);
773
774         end
775     endcase
776     end
777     /*END INSTRUCTION DECODE*/
778
779 end
780
781 endmodule

```

### 7.1.3 Chip8\_rand\_num\_generator.sv

```
1  /*
2  *
3  * Semi-naive pseudo-random number generator
4  *
5  * Implemented by Levi
6  *
7  */
8 module Chip8_rand_num_generator(input logic cpu_clk, output
9   logic [15:0] out);
10
11    logic [15:0] rand_num;
12
13  initial begin
14    rand_num <= 16'b1111010111010010;
15  end
16
17  always_ff @(posedge cpu_clk) begin
18    if(~| (rand_num[15:0])) begin
19      rand_num[15:0] <= 16'b1111010111010010;
20    end else begin
21      rand_num[0] <= rand_num[15] ^ rand_num[14];
22      rand_num[1] <= rand_num[14] ^ rand_num[13];
23      rand_num[2] <= rand_num[13] ^ rand_num[12];
24      rand_num[3] <= rand_num[12] ^ rand_num[11];
25      rand_num[4] <= rand_num[11] ^ rand_num[10];
26      rand_num[5] <= rand_num[10] ^ rand_num[9];
27      rand_num[6] <= rand_num[9] ^ rand_num[8];
28      rand_num[7] <= rand_num[8] ^ rand_num[7];
29      rand_num[8] <= rand_num[7] ^ rand_num[6];
30      rand_num[9] <= rand_num[6] ^ rand_num[5];
31      rand_num[10] <= rand_num[5] ^ rand_num[4];
32      rand_num[11] <= rand_num[4] ^ rand_num[3];
33      rand_num[12] <= rand_num[3] ^ rand_num[2];
34      rand_num[13] <= rand_num[2] ^ rand_num[1];
35      rand_num[14] <= rand_num[1] ^ rand_num[0];
36      rand_num[15] <= rand_num[0] ^ rand_num[15];
```

```

36         end
37
38         out <= rand_num;
39     end
40 endmodule

```

#### 7.1.4 Chip8\_Stack.sv

```

1  'include "../enums.svh"
2
3  module Chip8_Stack(
4      input logic          cpu_clk,      //clock
5      input logic          reset,       //reset
6      input STACK_OP       op,         //See enums.svh for
7      ~ ops
8      input logic [15:0]   writedata,   //input PC
9      output logic [15:0]  outdata,    //data output
10 );
11
12     logic [3:0] address = 4'd0;
13     logic [15:0] data;
14     logic          wren;
15     logic [15:0] q;
16
17     stack_ram stack(address, cpu_clk, data, wren, q);
18
19     logic[3:0]      stackptr = 4'h0;
20     logic          secondcycle = 1'h0;
21     logic          hold = 1'h0;
22
23     always_ff @(posedge cpu_clk) begin
24         if(reset) begin
25             address <= 4'd0;
26             hold <= 1'd0;
27             wren <= 1'b0;
28             secondcycle <= 1'h0;
29             stackptr <= 4'd0;

```

```

29     end else begin
30         case (op)
31             STACK_PUSH: begin
32                 if(~hold) begin
33                     address <= stackptr;
34                     data <= writedata;
35                     if(secondcycle == 1'h0) begin
36                         wren <= 1'h1;
37                         secondcycle <= 1'h1;
38                     end else begin
39                         wren <= 1'h0;
40                         stackptr <= stackptr + 4'b0001;
41                         secondcycle <= 1'h0;
42                         hold <= 1'b1;
43                     end
44                 end
45             end
46             STACK_POP: begin
47                 if(~hold) begin
48                     address <= stackptr - 4'b0001;
49                     wren <= 1'h0;
50                     outdata <= q;
51                     if(secondcycle == 1'h0) begin
52                         secondcycle <= 1'h1;
53                         stackptr <= stackptr - 4'b0001;
54                     end else begin
55                         secondcycle <= 1'h0;
56                         hold <= 1'b1;
57                     end
58                 end
59             end
60             STACK_HOLD: hold = 1'b0;
61             default : /* default */;
62         endcase
63     end
64 end
65 endmodule

```

### 7.1.5 Chip8\_ALU.sv

```
1  ****
2  * Chip8_ALU.sv
3  *
4  * Simple ALU supporting instructions:
5  *   - OR          - bitwise OR
6  *   - AND         - bitwise AND
7  *   - XOR         - bitwise XOR
8  *   - ADD         - Addition
9  *   - MINUS       - Subtract
10 *   - LSHIFT      - Shift left
11 *   - RSHIFT      - Shift right
12 *   - EQUALS      - Equals compare
13 *   - GREATER     - Greater than compare
14 *   - INC          - Increment
15 *
16 * This module is solely used by the Chip8_CPU module, and
17 * relies on the ALU_f
18 * enum defined in enums.svh
19 *
20 * AUTHORS: David Watkins, Ashley Kling
21 * Dependencies:
22 *   - enums.svh
23 *
24 'include "../enums.svh"
25
26 module Chip8_ALU(
27     input logic[15:0] input1, input2,
28     input ALU_f sel,
29
30     output logic[15:0] out,
31     output logic alu_carry);
32
33     logic[15:0] intermediate;
34
```

```

35      always_comb begin
36          case      (sel)
37
38              ALU_f_OR : begin
39                  alu_carry = 0;
40                  out = input1 | input2;
41              end
42
43              ALU_f_AND : begin
44                  alu_carry = 0;
45                  out = input1 & input2;
46              end
47
48              ALU_f_XOR : begin
49                  alu_carry = 0;
50                  out = input1 ^ input2;
51              end
52
53              ALU_f_ADD : begin
54                  out = input1 + input2;
55                  alu_carry = |(out[15:8]);
56              end
57
58              ALU_f_MINUS : begin
59                  alu_carry = input1 > input2;
60                  out = input1 - input2;
61              end
62
63              ALU_f_LSHIFT : begin
64                  alu_carry = 0;
65                  out = input1 << input2;
66              end
67
68              ALU_f_RSHIFT : begin
69                  alu_carry = 0;
70                  out = input1 >> input2;
71              end
72

```

```

73         ALU_f_EQUALS : begin
74             alu_carry = 0;
75             out = (input1 == input2);
76         end
77
78         ALU_f_GREATER : begin
79             alu_carry = 0;
80             out = (input1 > input2);
81         end
82
83         ALU_f_INC : begin
84             alu_carry = 0;
85             out = input1 + 1'h1;
86         end
87
88         default: begin
89             alu_carry = 0;
90             out = 0;
91         end
92     endcase
93 end
94

```

### 7.1.6 Chip8\_framebuffer.sv

```

1 ****
2 * Chip8_framebuffer.sv
3 *
4 * Top level framebuffer module that contains the memory for
5 * the main view
6 * and has wires into the VGA emulator to ouput video
7 *
8 * Built off of Stephen Edwards's VGA_LED code
9 *
10 * AUTHORS: David Watkins, Levi Oliver, Ashley Kling,
11 * Gabrielle Taylor
12 * Dependencies:

```

```

11  * - Chip8_VGA_Emulator.sv
12  * - Framebuffer.v
13
14  module Chip8_framebuffer(
15      input logic          clk,
16      input logic          reset,
17
18      input logic [4:0]    fb_addr_y, //max val = 31
19      input logic [5:0]    fb_addr_x, //max val = 63
20      input logic          fb_writedata, //data to write to
21      ↳      adresse.
22      input logic          fb_WE, //enable writing to address
23      input logic          is_paused,
24
25      output logic         fb_readdata, //data to write to
26      ↳      adresse.
27
28      output logic [7:0]   VGA_R, VGA_G, VGA_B,
29      output logic         VGA_CLK, VGA_HS, VGA_VS, VGA_BLANK_n,
30      output logic         VGA_SYNC_n
31  );
32
33
34
35
36  wire[10:0] fb_addr_general = (fb_addr_y << 6) +
37      ↳      (fb_addr_x);
38  wire[10:0] fb_addr_vga;
39  wire fb_writedata_general = fb_writedata;
40  wire fb_writedata_vga;
41  wire fb_WE_general = fb_WE;
42  wire fb_WE_vga = 1'b0; //vga emulator will never write to
43      ↳      FB mem
44  wire fb_readdata_general;
45  wire fb_readdata_vga;

```

```

44 assign fb_readdata = fb_readdata_general;
45
46 logic[10:0] counter;
47
48 initial begin
49     counter = 11'b0;
50 end
51
52 wire[10:0] copy_from_addr;
53 wire[10:0] copy_to_addr;
54 wire copy_data;
55 wire copyWE;
56 wire deadwire;
57
58 logic [31:0] fb_stage;
59 logic [31:0] time_since_last_copy;
60
61 initial begin
62     time_since_last_copy <= 32'h0;
63     fb_stage <= 32'h0;
64 end
65
66 always_ff @(posedge clk) begin
67     if(fb_WE) begin
68         fb_stage <= 32'h0;
69     end else if(fb_stage < FRAMEBUFFER_REFRESH_HOLD) begin
70         fb_stage <= fb_stage;
71     end else begin
72         fb_stage <= fb_stage + 32'h1;
73     end
74 end
75
76 always_ff @(posedge clk) begin
77     if(fb_stage >= FRAMEBUFFER_REFRESH_HOLD ||
78         time_since_last_copy > COPY_THRESHOLD) begin
79         counter <= counter + 1;
80
81         copyWE <= 1'b1;

```

```

81
82     copy_from_addr <= counter + 11'h1;
83     copy_to_addr <= counter;
84
85     if(counter == 11'b111_1111_1111)
86         → time_since_last_copy <= 32'h0;
87     end else begin
88         copyWE <= 1'b0;
89         time_since_last_copy <= time_since_last_copy +
90             → 32'h1;
91         counter <= 11'h0;
92     end
93 end
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116

```

```

copy_from_addr <= counter + 11'h1;
copy_to_addr <= counter;

if(counter == 11'b111_1111_1111)
    → time_since_last_copy <= 32'h0;
end else begin
    copyWE <= 1'b0;
    time_since_last_copy <= time_since_last_copy +
        → 32'h1;
    counter <= 11'h0;
end
end

Chip8_VGA_Emulator led_emulator(
    .clk50(clk),
    .reset(reset),
    .fb_pixel_data(fb_readdata_vga),
    .fb_request_addr(fb_addr_vga),
    .is_paused(is_paused),
    .VGA_R(VGA_R),
    .VGA_G(VGA_G),
    .VGA_B(VGA_B),
    .VGA_CLK(VGA_CLK),
    .VGA_HS(VGA_HS),
    .VGA_VS(VGA_VS),
    .VGA_BLANK_n(VGA_BLANK_n),
    .VGA_SYNC_n(VGA_SYNC_n)
);

Framebuffer from_cpu (
    .clock(clk),
    .address_a(fb_addr_general),
    .address_b(copy_from_addr),
    .data_a(fb_writedata_general),
    .data_b(fb_writedata_vga),
    .wren_a(fb_WE_general),
    .wren_b(fb_WE_vga),

```

```

117      .q_a(fb_readdata_general),
118      .q_b(copy_data)
119  );
120
121  Framebuffer toscreen (
122      .clock(clk),
123      .address_a(fb_addr_vga),
124      .address_b(copy_to_addr),
125      .data_a(fb_writedata_vga),
126      .data_b(copy_data),
127      .wren_a(fb_WE_vga),
128      .wren_b(copyWE),
129      .q_a(fb_readdata_vga),
130      .q_b(deadwire)
131  );
132
133
134 endmodule

```

### 7.1.7 Chip8\_VGA\_Emulator.sv

```

1  /*
2   * Chip8-Framebuffer to VGA module.
3   * Adjusts the dimensions of the screen so that it appears 8
4   * times larger.
5   *
6   * Developed by Levi and Ash
7   *
8   * Built off of Stephen Edwards's code
9   * Columbia University
10  */
11
11 module Chip8_VGA_Emulator(
12     input logic          clk50, reset,
13     //input logic [2047:0] framebuffer,
14     input logic fb_pixel_data,
15     input logic is_paused,

```

```

16   output logic[10:0] fb_request_addr,
17   output logic [7:0] VGA_R, VGA_G, VGA_B,
18   output logic      VGA_CLK, VGA_HS, VGA_VS, VGA_BLANK_n,
19   ↳ VGA_SYNC_n);
20
21 /*
22 * 640 X 480 VGA timing for a 50 MHz clock: one pixel
23 ↳ every other cycle
24 */
25 * HCOUNT 1599 0          1279          1599 0
26 *
27 *
28 * /SYNC/ BP /<-- HACTIVE -->/FP/SYNC/ BP /<-- HACTIVE
29 *
30 * /_____|           Video           |_____|           Video
31 */
32 // Parameters for hcount
33 parameter HACTIVE        = 11'd 1280,
34 HFRONT_PORCH = 11'd 32,
35 HSYNC         = 11'd 192,
36 HBACK_PORCH  = 11'd 96,
37 HTOTAL        = HACTIVE + HFRONT_PORCH + HSYNC +
38   ↳ HBACK_PORCH; // 1600
39
40 // Parameters for vcount
41 parameter VACTIVE        = 10'd 480,
42 VFRONT_PORCH = 10'd 10,
43 VSYNC         = 10'd 2,
44 VBACK_PORCH  = 10'd 33,
45 VTOTAL        = VACTIVE + VFRONT_PORCH + VSYNC +
46   ↳ VBACK_PORCH; // 525
47
48 logic [10:0] hcount; // Horizontal counter //
49 ↳ Hcount[10:1] indicates pixel column (0-639)
50 logic endOfLine;

```

```

49      always_ff @(posedge clk50 or posedge reset)
50          if (reset)           hcount <= 0;
51      else if (endOfLine)   hcount <= 0;
52          else                hcount <= hcount + 11'd 1;
53
54      assign endOfLine = hcount == HTOTAL - 1;
55
56      // Vertical counter
57      logic [9:0]             vcount;
58      logic                  endOfField;
59
60      always_ff @(posedge clk50 or posedge reset)
61          if (reset)           vcount <= 0;
62      else if (endOfLine)
63          if (endOfField)    vcount <= 0;
64          else                vcount <= vcount + 10'd 1;
65
66      assign endOfField = vcount == VTOTAL - 1;
67
68      // Horizontal sync: from 0x520 to 0x5DF (0x57F)
69      // 101 0010 0000 to 101 1101 1111
70      assign VGA_HS = !( (hcount[10:8] == 3'b101) &
71                      !(hcount[7:5] == 3'b111));
72      assign VGA_VS = !( vcount[9:1] == (VACTIVE + VFRONT_PORCH)
73                      / 2);
74
75      assign VGA_SYNC_n = 1; // For adding sync to video
76          // signals; not used for VGA
77
78      // Horizontal active: 0 to 1279      Vertical active: 0 to
79          // 479
80      // 101 0000 0000 1280              01 1110 0000 480
81      // 110 0011 1111 1599              10 0000 1100 524
82      assign VGA_BLANK_n = !( hcount[10] & (hcount[9] |
83                      hcount[8]) ) &
84          !( vcount[9] | (vcount[8:5] == 4'b1111) );
85
86      /* VGA_CLK is 25 MHz

```

```

82
83      *          -- / -- / -- / --
84      *
85      *
86      *      ----- / ----- /
87      */
88 assign VGA_CLK = hcount[0]; // 25 MHz clock: pixel
     ↳ latched on rising edge
89
90 parameter chip_hend = 7'd 64;
91 parameter chip_vend = 6'd 32;
92
93 parameter left_bound = 7'd64;
94 parameter right_bound = 10'd576;
95 parameter top_bound = 7'd112;
96 parameter bottom_bound = 9'd368;
97
98 logic[11:0] fb_pos;
99 assign fb_request_addr = (((vcnt[8:0] - top_bound) &
     ↳ (8'b1111_1000)) << (4'd3)) + ((hcount[10:1] -
     ↳ left_bound) >> (4'd3));
100
     //(((vcnt[8:0] - top_bound)
     ↳ >> (4'd3))*(7'd64)) +
     ↳ ((hcount[10:1] -
     ↳ left_bound) >> (4'd3));
101
     //this commented out line is
     ↳ the old code (which was
     ↳ believed to work). New
     ↳ code is more efficient
102
     //in area and time. Hopefully
     ↳ this helps if we have a
     ↳ too long critical path
103
104
105 logic inChip;
106 //120 <= Y-dim < 360
107 //64 <= X-dim < 576
108
```

```

109  /*
110  *      +-----+
111  *      | VGA Screen (640x480)          |
112  *      |   64                      576  |
113  *      |   +-----+112  |
114  *      |   |     Chip8 Screen    |   |
115  *      |   |     (64*8x32*8)    |   |
116  *      |   +-----+368  |
117  *      |
118  *      +-----+
119 */
120
121
122 //      assign inChip = (((hcount[10:1]) >= (chip_hend *
123 //                           ↳ (5'd8) + 7'd64)) & ((hcount[10:1]) < (chip_hend * (5'd8)
124 //                           + 10'd576)) &
125 //                           ((vcount[8:0]) >= (chip_vend * (5'd8) +
126 //                           ↳ 7'd112)) & ((vcount[8:0]) < (chip_vend * (5'd8) +
127 //                           + 10'd368));
128
129
130
131
132 /**
133 * 16 columns, 8 rows
134 * +-----+
135 * /      px24          |
136 * /      px48      === === = = === === px432  |
137 * /      = = = = = = = = == == |
138 * /      === === = = === === = = |
139 * /      = = = = = = = = == == |
140 * /      = = = === === === === px88  |
141 */

```

```

142      */
143
144      parameter paused_left = 10'd64;
145      parameter paused_right = 10'd576;
146      parameter paused_top = 10'd24;
147      parameter paused_bottom = 10'd88;
148
149      logic [9:0] hcount_offseted, vcount_offseted;
150      assign hcount_offseted = (hcount[10:1] - paused_left)
151          ↳ >> 4;
152      assign vcount_offseted = (vcount[8:0] - paused_top) >>
153          ↳ 3;
154
155      reg [31:0] romdata [7:0];
156
157      initial begin
158          romdata[7] =
159              ↳ 32'b0000_1110_1110_1010_1110_1110_1110_0000;
160          romdata[6] =
161              ↳ 32'b0000_1010_1010_1010_1000_1000_1011_0000;
162          romdata[5] =
163              ↳ 32'b0000_1110_1110_1010_1110_1110_1001_0000;
164          romdata[4] =
165              ↳ 32'b0000_1000_1010_1010_0010_1000_1011_0000;
166          romdata[3] =
167              ↳ 32'b0000_1000_1010_1110_1110_1110_1110_0000;
168          romdata[2] = 32'b0;
169          romdata[1] = 32'b0;
170          romdata[0] = 32'b0;
171      end
172      assign inPaused = is_paused & (
173          ((hcount[10:1]) >= (paused_left)) &
174          ((hcount[10:1]) < (paused_right)) &
175          ((vcount[8:0]) >= (paused_top)) &
176          ((vcount[8:0]) < (paused_bottom)) &
177          romdata[vcount_offseted] [hcount_offseted]
178      );

```

```

173    always_comb begin
174        {VGA_R, VGA_G, VGA_B} = {8'h0, 8'h0, 8'h0}; // 
175        ↪ Black
176        if (inChip & fb_pixel_data)
177            ↪ begin//framebuffer[fb_pos]) begin
178                //White to show on-pixel
179                {VGA_R, VGA_G, VGA_B} = {8'hFF, 8'hFF, 8'hFF};
180            end else if(inChip) begin
181                //purple to show general area
182                {VGA_R, VGA_G, VGA_B} = {8'h0, 8'h0, 8'hFF};
183            end else if(inPaused) begin
184                {VGA_R, VGA_G, VGA_B} = {8'hFF, 8'hFF, 8'hFF};
185            end
186        end
187    endmodule // VGA_LED_Emulator

```

### 7.1.8 audio\_codec.sv

```

1 // Original audio codec code taken from
2 //Howard Mao's FPGA blog
3 //http://zhehaomao.com/blog/fpga/2014/01/15/sockit-8.html
4 //Modified as needed

5
6 /*
7 audio_codec.sv
8 Sends samples to the audio codec ssm 2603 at audio clock
9     → rate.
10 */
11
12 //Audio codec interface
13 module audio_codec (
14     input  clk,
15     input  reset,
16     output [1:0] sample_end,
17     output [1:0] sample_req,

```

```

18     input  [15:0] audio_output,
19     output [15:0] audio_input,
20     // 1 - left, 0 - right
21     input  [1:0] channel_sel,
22
23     output AUD_ADCLRCK,
24     input  AUD_ADCDAT,
25     output AUD_DACLRCK,
26     output AUD_DACDAT,
27     output AUD_BCLK
28 );
29
30 reg [7:0] lrck_divider;
31 reg [1:0] bclk_divider;
32
33 reg [15:0] shift_out;
34 reg [15:0] shift_temp;
35 reg [15:0] shift_in;
36
37 wire lrck = !lrck_divider[7];
38
39 assign AUD_ADCLRCK = lrck;
40 assign AUD_DACLRCK = lrck;
41 assign AUD_BCLK = bclk_divider[1];
42 assign AUD_DACDAT = shift_out[15];
43
44 always @ (posedge clk) begin
45     if (reset) begin
46         lrck_divider <= 8'hff;
47         bclk_divider <= 2'b11;
48     end else begin
49         lrck_divider <= lrck_divider + 1'b1;
50         bclk_divider <= bclk_divider + 1'b1;
51     end
52 end
53
54 assign sample_end[1] = (lrck_divider == 8'h40);
55 assign sample_end[0] = (lrck_divider == 8'hc0);

```

```

56 assign audio_input = shift_in;
57 assign sample_req[1] = (lrck_divider == 8'hfe);
58 assign sample_req[0] = (lrck_divider == 8'h7e);
59
60 wire clr_lrck = (lrck_divider == 8'h7f);
61 wire set_lrck = (lrck_divider == 8'hff);
62 // high right after bclk is set
63 wire set_bclk = (bclk_divider == 2'b10 && !lrck_divider[6]);
64 // high right before bclk is cleared
65 wire clr_bclk = (bclk_divider == 2'b11 && !lrck_divider[6]);
66
67 always @ (posedge clk) begin
68   if (reset) begin
69     shift_out <= 16'h0;
70     shift_in <= 16'h0;
71     shift_in <= 16'h0;
72   end else if (set_lrck || clr_lrck) begin
73     // check if current channel is selected
74     if (channel_sel[set_lrck]) begin
75       shift_out <= audio_output;
76       shift_temp <= audio_output;
77       shift_in <= 16'h0;
78     // repeat the sample from the other channel if not
79     end else shift_out <= shift_temp;
80   end else if (set_bclk == 1) begin
81     // only read in if channel is selected
82     if (channel_sel[lrck])
83       shift_in <= {shift_in[14:0], AUD_ADCDAT};
84   end else if (clr_bclk == 1) begin
85     shift_out <= {shift_out[14:0], 1'b0};
86   end
87 end
88
89 endmodule

```

### 7.1.9 audio\_effects.sv

```
1 //Original audio codec code taken from
2 //Howard Mao's FPGA blog
3 //http://zhehaomao.com/blog/fpga/2014/01/15/sockit-8.html
4 //Modified as needed

5
6 /* audio_effects.sv
7     Sends hardcoded beep samples to audio codec interface
8 */
9
10 module audio_effects (
11     input  audio_clk,
12     input  main_clk,
13     input  reset,
14
15     input  sample_end,
16     input  sample_req,
17
18     output [15:0] audio_output,
19     input  [15:0] audio_input,
20
21     input  control
22 );
23
24 reg [15:0] romdata [0:99];
25 reg [6:0] index = 7'd0;
26 reg [15:0] last_sample;
27 reg [15:0] dat;
28 wire [15:0] filter_output;
29 wire filter_finish;
30
31 assign audio_output = dat;
32
33 parameter SINE      = 0;
34 parameter FEEDBACK = 1;
35 parameter FILTER    = 2;
36
```

```

37 | parameter SINE_LAST = 7'd99;
38 |
39 | initial begin
40 |     romdata[0] = 16'h0000;
41 |     romdata[1] = 16'h0805;
42 |     romdata[2] = 16'h1002;
43 |     romdata[3] = 16'h17ee;
44 |     romdata[4] = 16'h1fc3;
45 |     romdata[5] = 16'h2777;
46 |     romdata[6] = 16'h2f04;
47 |     romdata[7] = 16'h3662;
48 |     romdata[8] = 16'h3d89;
49 |     romdata[9] = 16'h4472;
50 |     romdata[10] = 16'h4b16;
51 |     romdata[11] = 16'h516f;
52 |     romdata[12] = 16'h5776;
53 |     romdata[13] = 16'h5d25;
54 |     romdata[14] = 16'h6276;
55 |     romdata[15] = 16'h6764;
56 |     romdata[16] = 16'h6bea;
57 |     romdata[17] = 16'h7004;
58 |     romdata[18] = 16'h73ad;
59 |     romdata[19] = 16'h76e1;
60 |     romdata[20] = 16'h799e;
61 |     romdata[21] = 16'h7be1;
62 |     romdata[22] = 16'h7da7;
63 |     romdata[23] = 16'h7eef;
64 |     romdata[24] = 16'h7fb7;
65 |     romdata[25] = 16'h7fff;
66 |     romdata[26] = 16'h7fc6;
67 |     romdata[27] = 16'h7f0c;
68 |     romdata[28] = 16'h7dd3;
69 |     romdata[29] = 16'h7c1b;
70 |     romdata[30] = 16'h79e6;
71 |     romdata[31] = 16'h7737;
72 |     romdata[32] = 16'h7410;
73 |     romdata[33] = 16'h7074;
74 |     romdata[34] = 16'h6c67;

```

```
75     romdata[35] = 16'h67ed;
76     romdata[36] = 16'h630a;
77     romdata[37] = 16'h5dc4;
78     romdata[38] = 16'h5820;
79     romdata[39] = 16'h5222;
80     romdata[40] = 16'h4bd3;
81     romdata[41] = 16'h4537;
82     romdata[42] = 16'h3e55;
83     romdata[43] = 16'h3735;
84     romdata[44] = 16'h2fdd;
85     romdata[45] = 16'h2855;
86     romdata[46] = 16'h20a5;
87     romdata[47] = 16'h18d3;
88     romdata[48] = 16'h10e9;
89     romdata[49] = 16'h08ee;
90     romdata[50] = 16'h00e9;
91     romdata[51] = 16'hf8e4;
92     romdata[52] = 16'hf0e6;
93     romdata[53] = 16'he8f7;
94     romdata[54] = 16'he120;
95     romdata[55] = 16'hd967;
96     romdata[56] = 16'hd1d5;
97     romdata[57] = 16'hca72;
98     romdata[58] = 16'hc344;
99     romdata[59] = 16'hbc54;
100    romdata[60] = 16'hb5a7;
101    romdata[61] = 16'haf46;
102    romdata[62] = 16'ha935;
103    romdata[63] = 16'ha37c;
104    romdata[64] = 16'h9e20;
105    romdata[65] = 16'h9926;
106    romdata[66] = 16'h9494;
107    romdata[67] = 16'h906e;
108    romdata[68] = 16'h8cb8;
109    romdata[69] = 16'h8976;
110    romdata[70] = 16'h86ab;
111    romdata[71] = 16'h845a;
112    romdata[72] = 16'h8286;
```

```

113     romdata[73] = 16'h8130;
114     romdata[74] = 16'h8059;
115     romdata[75] = 16'h8003;
116     romdata[76] = 16'h802d;
117     romdata[77] = 16'h80d8;
118     romdata[78] = 16'h8203;
119     romdata[79] = 16'h83ad;
120     romdata[80] = 16'h85d3;
121     romdata[81] = 16'h8875;
122     romdata[82] = 16'h8b8f;
123     romdata[83] = 16'h8f1d;
124     romdata[84] = 16'h931e;
125     romdata[85] = 16'h978c;
126     romdata[86] = 16'h9c63;
127     romdata[87] = 16'ha19e;
128     romdata[88] = 16'ha738;
129     romdata[89] = 16'had2b;
130     romdata[90] = 16'hb372;
131     romdata[91] = 16'hba05;
132     romdata[92] = 16'hc0df;
133     romdata[93] = 16'hc7f9;
134     romdata[94] = 16'hcf4b;
135     romdata[95] = 16'hd6ce;
136     romdata[96] = 16'hde7a;
137     romdata[97] = 16'he648;
138     romdata[98] = 16'hee30;
139     romdata[99] = 16'hf629;
140 end
141
142 always @(*) begin
143
144     if (control)
145         dat <= romdata[index];
146     else
147         dat <= 16'd0;
148 end
149
150 always @(posedge audio_clk) begin

```

```

151     if (sample_req) begin
152         if (index == SINE_LAST)
153             index <= 7'd00;
154         else
155             index <= index + 1'b1;
156     end
157 end
158
159
160 endmodule

```

### 7.1.10 Chip8\_SoundController.sv

```

1  ****
2  * Chip8_SoundController.sv
3  *
4  * Top level controller for audio output on the Chip8
5  * Designed to interact directly with a Altera SoCKit
6  * Cyclone V board
7  * Source originally borrowed from:
8  * Howard Mao's FPGA blog
9  * http://zhehaomao.com/blog/fpga/2014/01/15/sockit-8.html
10 *
11 * AUTHORS: David Watkins, Gabrielle Taylor
12 * Dependencies:
13 * - Chip8_Sound/clock_pll.v
14 * - Chip8_Sound/audio_effects.sv
15 * - Chip8_Sound/i2c_av_config.sv
16 * - Chip8_Sound/audio_codec.sv
17
18 module Chip8_SoundController (
19     input  OSC_50_B8A,    //reference clock
20     inout AUD_ADCLRCK, //Channel clock for ADC
21     input  AUD_ADCDAT,
22     inout AUD_DACLRCK, //Channel clock for DAC

```

```

23     output AUD_DACDAT, //DAC data
24     output AUD_XCK,
25     inout AUD_BCLK, // Bit clock
26     output AUD_I2C_SCLK, //I2C clock
27     inout AUD_I2C_SDAT, //I2C data
28     output AUD_MUTE, //Audio mute
29
30     input logic clk,
31     input logic is_on, //Turn on the output
32     input logic reset
33 );
34
35 wire main_clk;
36 wire audio_clk;
37
38 wire [1:0] sample_end;
39 wire [1:0] sample_req;
40 wire [15:0] audio_output;
41 wire [15:0] audio_input;
42
43 wire [3:0] status;
44
45 clock_pll pll (
46     .refclk (OSC_50_B8A),
47     .rst (reset),
48     .outclk_0 (audio_clk),
49     .outclk_1 (main_clk)
50 );
51
52 i2c_av_config av_config (
53     .clk (main_clk),
54     .reset (reset),
55     .i2c_sclk (AUD_I2C_SCLK),
56     .i2c_sdat (AUD_I2C_SDAT),
57     .status (status)
58 );
59
60 assign AUD_XCK = audio_clk;

```

```

61 assign AUD_MUTE = is_on;
62
63 audio_codec ac (
64     .clk (audio_clk),
65     .reset (reset),
66     .sample_end (sample_end),
67     .sample_req (sample_req),
68     .audio_output (audio_output),
69     .audio_input (audio_input),
70     .channel_sel (2'b10),
71
72     .AUD_ADCLRCK (AUD_ADCLRCK),
73     .AUD_ADCDAT (AUD_ADCDAT),
74     .AUD_DACLRCK (AUD_DACLRCK),
75     .AUD_DACDAT (AUD_DACDAT),
76     .AUD_BCLK (AUD_BCLK)
77 );
78
79 audio_effects ae (
80     .audio_clk (audio_clk),
81     .main_clk (main_clk),
82     .sample_end (sample_end[1]),
83     .sample_req (sample_req[1]),
84     .audio_output (audio_output),
85     .audio_input (audio_input),
86     .control (1'b1)
87 );
88
89 endmodule

```

### 7.1.11 i2c\_av\_config.sv

```

1 // Original audio codec code taken from
2 //Howard Mao's FPGA blog
3 //http://zhehaomao.com/blog/fpga/2014/01/15/sockit-8.html
4 //Modified as needed
5

```

```

6 //configure Audio codec using the I2C protocol
7 module i2c_av_config (
8     input clk,
9     input reset,
10
11     output i2c_sclk,
12     inout i2c_sdat,
13
14     output [3:0] status
15 );
16
17 reg [23:0] i2c_data;
18 reg [15:0] lut_data;
19 reg [3:0] lut_index = 4'd0;
20
21 parameter LAST_INDEX = 4'ha;
22
23 reg i2c_start = 1'b0;
24 wire i2c_done;
25 wire i2c_ack;
26
27 i2c_controller control (
28     .clk (clk),
29     .i2c_sclk (i2c_sclk),
30     .i2c_sdat (i2c_sdat),
31     .i2c_data (i2c_data),
32     .start (i2c_start),
33     .done (i2c_done),
34     .ack (i2c_ack)
35 );
36
37 always @(*) begin
38     case (lut_index)
39         4'h0: lut_data <= 16'h0c10; // power on everything
40             ↳ except out
41         4'h1: lut_data <= 16'h0017; // left input
42         4'h2: lut_data <= 16'h0217; // right input
43         4'h3: lut_data <= 16'h0479; // left output

```

```

43      4'h4: lut_data <= 16'h0679; // right output
44      4'h5: lut_data <= 16'h08d4; // analog path
45      4'h6: lut_data <= 16'h0a04; // digital path
46      4'h7: lut_data <= 16'h0e01; // digital IF
47      4'h8: lut_data <= 16'h1020; // sampling rate
48      4'h9: lut_data <= 16'h0c00; // power on everything
49      4'ha: lut_data <= 16'h1201; // activate
50      default: lut_data <= 16'h0000;
51  endcase
52 end
53
54 reg [1:0] control_state = 2'b00;
55
56 assign status = lut_index;
57
58 always @ (posedge clk) begin
59   if (reset) begin
60     lut_index <= 4'd0;
61     i2c_start <= 1'b0;
62     control_state <= 2'b00;
63   end else begin
64     case (control_state)
65       2'b00: begin
66         i2c_start <= 1'b1;
67         i2c_data <= {8'h34, lut_data};
68         control_state <= 2'b01;
69       end
70       2'b01: begin
71         i2c_start <= 1'b0;
72         control_state <= 2'b10;
73       end
74       2'b10: if (i2c_done) begin
75         if (i2c_ack) begin
76           if (lut_index == LAST_INDEX)
77             control_state <= 2'b11;
78           else begin
79             lut_index <= lut_index + 1'b1;
80             control_state <= 2'b00;

```

```

81           end
82       end else
83           control_state <= 2'b00;
84       end
85   endcase
86 end
87
88
89 endmodule

```

### 7.1.12 i2c\_controller.sv

```

1 // Original audio codec code taken from
2 //Howard Mao's FPGA blog
3 //http://zhehaomao.com/blog/fpga/2014/01/15/sockit-8.html
4 //Modified as needed
5
6 //implement the I2C protocol to configure registers in ssm
7 //→ 2603 audio codec
8 module i2c_controller (
9     input  clk,
10    output i2c_sclk,      //i2c clock
11    inout  i2c_sdat,      //i2c data  out
12
13    input  start,
14    output done,
15    output ack,
16
17    input [23:0] i2c_data
18 );
19
20 reg [23:0] data;
21
22 reg [4:0] stage;
23 reg [6:0] sclk_divider;
24 reg clock_en = 1'b0;

```

```

25
26 // don't toggle the clock unless we're sending data
27 // clock will also be kept high when sending START and STOP
28 //    ↪ symbols
29 assign i2c_sclk = (!clock_en) || sclk_divider[6];
30 wire midlow = (sclk_divider == 7'h1f);
31
32 reg sdat = 1'b1;
33 // rely on pull-up resistor to set SDAT high
34 assign i2c_sdat = (sdat) ? 1'b1 : 1'b0;
35
36 reg [2:0] acks;
37
38 parameter LAST_STAGE = 5'd29;
39
40 assign ack = (acks == 3'b000);
41 assign done = (stage == LAST_STAGE);
42
43 //implementing I2C protocol
44 always @(posedge clk) begin
45     if (start) begin
46         sclk_divider <= 7'd0;
47         stage <= 5'd0;
48         clock_en = 1'b0;
49         sdat <= 1'b1;
50         acks <= 3'b111;
51         data <= i2c_data;
52     end else begin
53         if (sclk_divider == 7'd127) begin
54             sclk_divider <= 7'd0;
55
56             if (stage != LAST_STAGE)
57                 stage <= stage + 1'b1;
58
59             case (stage)
60                 // after start
61                 5'd0:  clock_en <= 1'b1;

```

```

62      // receive acks
63      5'd9:  acks[0] <= i2c_sdat;
64      5'd18: acks[1] <= i2c_sdat;
65      5'd27: acks[2] <= i2c_sdat;
66      // before stop
67      5'd28: clock_en <= 1'b0;
68      endcase
69  end else
70      sclk_divider <= sclk_divider + 1'b1;
71
72  if (midlow) begin
73      case (stage)
74          // start
75          5'd0: sdat <= 1'b0;
76          // byte 1
77          5'd1: sdat <= data[23];
78          5'd2: sdat <= data[22];
79          5'd3: sdat <= data[21];
80          5'd4: sdat <= data[20];
81          5'd5: sdat <= data[19];
82          5'd6: sdat <= data[18];
83          5'd7: sdat <= data[17];
84          5'd8: sdat <= data[16];
85          // ack 1
86          5'd9: sdat <= 1'b1;
87          // byte 2
88          5'd10: sdat <= data[15];
89          5'd11: sdat <= data[14];
90          5'd12: sdat <= data[13];
91          5'd13: sdat <= data[12];
92          5'd14: sdat <= data[11];
93          5'd15: sdat <= data[10];
94          5'd16: sdat <= data[9];
95          5'd17: sdat <= data[8];
96          // ack 2
97          5'd18: sdat <= 1'b1;
98          // byte 3
99          5'd19: sdat <= data[7];

```

```

100      5'd20: sdat <= data[6];
101      5'd21: sdat <= data[5];
102      5'd22: sdat <= data[4];
103      5'd23: sdat <= data[3];
104      5'd24: sdat <= data[2];
105      5'd25: sdat <= data[1];
106      5'd26: sdat <= data[0];
107      // ack 3
108      5'd27: sdat <= 1'b1;
109      // stop
110      5'd28: sdat <= 1'b0;
111      5'd29: sdat <= 1'b1;
112      endcase
113  end
114 end
115
116
117 endmodule

```

### 7.1.13 clk\_div.sv

```

1 ****
2 * clk_div.sv
3 *
4 * A clock divider module for outputting a 60 Hz clock from
5 * a 50 MHz clock
6 *
7 * AUTHORS: Gabrielle Taylor
8 * Dependencies:
9 ****
10 module clk_div (
11     input logic clk_in,
12     input logic reset,    //resets on high
13     output logic clk_out);
14

```

```

15 //Input: 50 MHz clock
16 //Output: 60 Hz clock
17 //NOTE - Actual output clock frequency: 60.000024 Hz
18 //Calculation:
19 //50 MHz = 50,000,000 hz
20 //50,000,000 hz / 60 hz = 833,333.33
21 //Scaling factor rounded to 833,333

22
23 logic [19:0] count = 20'd0;           //counts up to
24   ↳ 833333
25 logic [19:0] stop = 20'd833333;      //833333 in hex
26
27 always @(posedge clk_in)
28 begin
29   if(~reset) begin
30     clk_out <= 1'b0;
31     if (count==stop) begin
32       count <= 20'd0;
33       clk_out <= 1'b1; //set clock high for one 60
34         ↳ MHz cycle
35     end else begin
36       count <= count + 20'd1;
37     end
38   end else begin
39     count <= 20'd0;
40     clk_out <= 1'b0;
41   end
42 end
43
44 endmodule

```

### 7.1.14 timer.sv

```

1 ****
2 * timer.sv
3 *
4 * Module for outputting a count down value based on a 60 Hz
5   ↳ clock

```

```

5   * Used for both the sound_timer and delay_timer in the
6   * Chip8_Top module
7   *
8   * AUTHORS: David Watkins, Gabrielle Taylor
9   * Dependencies:
10
11 module timer (
12     input logic write_enable,           //Write enable
13     input logic clk,                 //50 MHz clock
14     input logic clk_60,              //60 Hz clock
15     input logic [7:0] data,
16     output logic out,
17     output logic [7:0] output_data);
18
19     logic [7:0] delay_reg = 8'b0000_0000;
20
21     always @ (posedge clk)
22     begin
23         if (write_enable) begin
24             delay_reg <= data;
25         end else if (clk_60 & (~delay_reg)) begin
26             delay_reg <= delay_reg - 8'd1;
27         end
28         out <= ~delay_reg;
29         output_data <= delay_reg;
30     end
31
32 endmodule

```

### 7.1.15 Chip8.Top.sv

```

1   /*
2   * Chip8.Top.sv
3   *

```

```

4   * Top level Chip8 module that controls all other modules
5   *
6   * AUTHORS: David Watkins, Levi Oliver, Ashley Kling,
7   *          Gabriele Taylor
8   * Dependencies:
9   * - Chip8_SoundController.sv
10  * - Chip8_framebuffer.sv
11  * - timer.sv
12  * - clk_div.sv
13  * - memory.v
14  * - reg_file.v
15  * - enums.svh
16  * - utils.svh
17  * - Chip8_CPU.sv
18
19 'include "enums.svh"
20 'include "utils.svh"
21
22 module Chip8_Top(
23     input logic      clk,
24     input logic      reset,
25     input logic [31:0] writedata,
26     input logic      write,
27     input             chipselect,
28     input logic [17:0] address,
29
30     output logic [31:0] data_out,
31
32     //VGA Output
33     output logic [7:0]  VGA_R, VGA_G, VGA_B,
34     output logic        VGA_CLK, VGA_HS, VGA_VS, VGA_BLANK_n,
35     output logic        VGA_SYNC_n,
36
37     //Audio Output
38     input  OSC_50_B8A,      //reference clock
39     inout AUD_ADCLRCK,    //Channel clock for ADC

```

```

40      input  AUD_ADCDAT,
41      inout AUD_DACLRCK,      //Channel clock for DAC
42      output AUD_DACDAT,     //DAC data
43      output AUD_XCK,
44      inout AUD_BCLK,        //Bit clock
45      output AUD_I2C_SCLK,   //I2C clock
46      inout AUD_I2C_SDAT,    //I2C data
47      output AUD_MUTE       //Audio mute
48 );
49
50 //Index register
51 logic [15:0] I;
52
53 //Program counter
54 logic [7:0] pc_state;
55 logic [11:0] pc = 12'h200;
56 logic [11:0] next_pc;
57 logic [31:0] stage;
58 logic      halt_for_keypress;
59 logic [31:0] last_stage;
60 //Framebuffer values
61 logic      fbreset;
62 logic [4:0]  fb_addr_y;//max val = 31
63 logic [5:0]  fb_addr_x;//max val = 63
64 logic      fb_writedata; //data to write to addresse.
65 logic      fb_WE; //enable writing to address
66 logic      fb_readdata; //data to write to addresse.
67 logic      fb_paused;
68
69 //Keyboard
70 logic      ispressed;
71 logic [3:0] key;
72
73 //Memory
74 logic [7:0] memwritedata1, memwritedata2;
75 logic      memWE1, memWE2;
76 logic [11:0] memaddr1, memaddr2;
77 logic [7:0] memreaddata1, memreaddata2;

```

```

78
79 //Reg file
80 logic [7:0] reg_writedata1, reg_writedata2;
81 logic      regWE1, regWE2;
82 logic [3:0] reg_addr1, reg_addr2;
83 logic [7:0] reg_readdata1, reg_readdata2;
84
85 //CPU
86 logic [15:0] cpu_instruction;
87 logic      cpu_delay_timer_WE, cpu_sound_timer_WE;
88 logic [7:0]  cpu_delay_timer_writedata,
    ↳  cpu_sound_timer_writedata;
89 PC_SRC      cpu_pc_src;
90 logic [11:0] cpu_PC_writedata;
91 logic      cpu_reg_WE1, cpu_reg_WE2;
92 logic [3:0]  cpu_reg_addr1, cpu_reg_addr2;
93 logic [7:0]  cpu_reg_writedata1, cpu_reg_writedata2;
94 logic      cpu_mem_WE1, cpu_mem_WE2;
95 logic [11:0] cpu_mem_addr1, cpu_mem_addr2;
96 logic      cpu_mem_request;
97 logic [7:0]  cpu_mem_writedata1, cpu_mem_writedata2;
98 logic      cpu_reg_I_WE;
99 logic [15:0] cpu_reg_I_writedata;
100 logic     cpu_fbreset;
101 logic [4:0]  cpu_fb_addr_y;
102 logic [5:0]  cpu_fb_addr_x;
103 logic      cpu_fb_writedata;
104 logic      cpu_fb_readdata;
105 logic      cpu_fb_WE;
106 logic      cpu_halt_for_keypress;
107 logic      cpu_stk_reset;
108 STACK_OP   cpu_stk_op;
109 logic[15:0]  cpu_stk_writedata;
110 logic      cpu_bit_overwritten;
111 logic      cpu_is_drawing;
112
113 //Sound
114 logic sound_on;

```

```

115 logic sound_reset;
116
117 //Timers
118 logic      clk_div_reset, clk_div_clk_out;
119 logic      delay_timer_write_enable, delay_timer_out;
120 logic [7:0] delay_timer_data, delay_timer_output_data;
121 logic      sound_timer_write_enable, sound_timer_out;
122 logic [7:0] sound_timer_data, sound_timer_output_data;
123
124 //Stack
125 logic      stack_reset;
126 STACK_OP   stack_op;
127 logic [15:0] stack_writedata;
128 logic [15:0] stack_outdata;
129
130 //State
131 Chip8_STATE state = Chip8_PAUSED;
132 logic      bit_overwritten;
133 logic      is_drawing;
134
135 //ARM Registers
136 logic [5:0] fbvx_prev;
137 logic [4:0] fbvy_prev;
138 logic [11:0] mem_addr_prev;
139 logic      chipselect_happened;
140
141 //Chipselect temporary values
142 logic [3:0] chip_reg_addr1_prev;
143 logic      chip_regWE1_prev;
144 logic [15:0] chip_reg_writedata1_prev;
145 logic [5:0] chip_fb_addr_x_prev;
146 logic [4:0] chip_fb_addr_y_prev;
147 logic      chip_fb_writedata_prev;
148 logic      chip_fb_WE_prev;
149 logic [11:0] chip_memaddr1_prev;
150 logic      chip_memWE1_prev;
151 logic [7:0] chip_memwritedata1_prev;
152 logic      chip_sound_timer_write_enable_prev;

```

```

153      logic      chip_delay_timer_write_enable_prev;
154
155      initial begin
156          pc <= 12'h200;
157
158          last_stage <= 32'h0;
159          memWE1 <= 1'b0;
160          memWE2 <= 1'b0;
161          cpu_instruction <= 16'h0;
162          delay_timer_write_enable <= 1'b0;
163          sound_timer_write_enable <= 1'b0;
164          I <= 16'h0;
165          regWE1 <= 1'b0;
166          regWE2 <= 1'b0;
167
168          fbreset <= 1'b0;
169          fb_addr_y <= 5'b0;
170          fb_addr_x <= 6'b0;
171          fb_writedata <= 1'b0;
172          fb_WE <= 1'b0;
173
174          stack_op <= STACK_HOLD;
175
176          state <= Chip8_PAUSED;
177          cpu_instruction <= 16'h0;
178          stage <= 32'h0;
179
180          stack_reset <= 1'b1;
181
182          fbvx_prev <= 6'h0;
183          fbvy_prev <= 5'h0;
184          mem_addr_prev <= 12'h0;
185
186          sound_on <= 1'b0;
187          chipselect_happened <= 1'b0;
188
189          fb_paused <= 1'b1;
190

```

```

191      halt_for_keypress <= 1'b0;
192  end
193
194  always_ff @(posedge clk) begin
195    if(reset) begin
196      //Add initial values for code
197      pc <= 12'h200;
198
199      memWE1 <= 1'b0;
200      memWE2 <= 1'b0;
201      cpu_instruction <= 16'h0;
202      delay_timer_write_enable <= 1'b0;
203      sound_timer_write_enable <= 1'b0;
204      I <= 16'h0;
205      regWE1 <= 1'b0;
206      regWE2 <= 1'b0;
207
208      fbreset <= 1'b0;
209      fb_addr_y <= 5'b0;
210      fb_addr_x <= 6'b0;
211      fb_writedata <= 1'b0;
212      fb_WE <= 1'b0;
213
214      stack_op <= STACK_HOLD;
215
216      state <= Chip8_PAUSED;
217      cpu_instruction <= 16'h0;
218      stage <= 32'h0;
219
220      stack_reset <= 1'b1;
221
222      fbvx_prev <= 6'h0;
223      fbvy_prev <= 5'h0;
224      mem_addr_prev <= 12'h0;
225
226      sound_on <= 1'b0;
227      chipselect_happened <= 1'b0;
228

```

```

229     fb_paused <= 1'b1;
230
231     halt_for_keypress <= 1'b0;
232
233     //Handle input from the ARM processor
234 end else if(chipselect) begin
235
236     chipselect_happened <= 1'b1;
237
238     if(~chipselect_happened) begin
239         chip_sound_timer_write_enable_prev <=
240             → sound_timer_write_enable;
241         chip_delay_timer_write_enable_prev <=
242             → delay_timer_write_enable;
243         chip_reg_addr1_prev <= reg_addr1;
244         chip_regWE1_prev <= regWE1;
245         chip_reg_writedata1_prev <= reg_writedata1;
246         chip_fb_addr_x_prev <= fb_addr_x;
247         chip_fb_addr_y_prev <= fb_addr_y;
248         chip_fb_writedata_prev <= fb_writedata;
249         chip_fb_WE_prev <= fb_WE;
250         chip_memaddr1_prev <= memaddr1;
251         chip_memWE1_prev <= memWE1;
252         chip_memwritedata1_prev <= memwritedata1;
253     end
254
255     casex (address)
256
257         //Read/write from register
258         18'b00_0000_0000_0000_xxxx: begin
259             reg_addr1 <= address[3:0];
260             data_out <= {24'h0, reg_readdata1};
261             if(write) begin
262                 regWE1 <= 1'b1;
263                 reg_writedata1 <= writedata[7:0];
264             end
265         end

```

```

265      18'h10 : begin
266          if(write)
267              I <= writedata[15:0];
268              data_out <= {16'b0, I};
269      end
270
271      //Read/write to sound_timer
272      18'h11 : begin
273          data_out <= {24'h0,
274              → sound_timer_output_data};
275          if(write) begin
276              sound_timer_write_enable <= 1'b1;
277              sound_timer_data <= writedata[7:0];
278          end
279      end
280
281      //Read/write to delay_timer
282      18'h12 : begin
283          data_out <= {24'h0,
284              → delay_timer_output_data};
285          if(write) begin
286              delay_timer_write_enable <= 1'b1;
287              delay_timer_data <= writedata[7:0];
288          end
289      end
290
291      //Reset stack
292      18'h13 : begin
293          if(write) stack_reset <= 1'b1;
294          data_out <= 32'h13;
295      end
296
297      //Read/write to program counter
298      18'h14 : begin
299          data_out <= {4'h0, cpu_instruction, pc};
300          if(write) pc <= writedata[11:0]; //0-4095
301      end

```

```

301 //Read/write key presses
302 18'h15 : begin
303     data_out <= {27'b0, ispressed, key};
304     if(write) begin
305         ispressed <= writedata[4];
306         key <= writedata[3:0];
307     end
308 end
309
310 //Read/write the state of the emulator
311 18'h16 : begin
312     data_out <= state;
313     if(write) begin
314         case (writedata[1:0])
315             2'h0: state <= Chip8_RUNNING;
316             2'h1: state <=
317                 → Chip8_RUN_INSTRUCTION;
318             2'h2: state <= Chip8_PAUSED;
319             default : state <= Chip8_PAUSED;
320         endcase
321     end
322 end
323
324 //Modify framebuffer
325 18'h17 : begin
326     if(write) begin
327         fbvx_prev <= writedata[10:5];
328         fbvy_prev <= writedata[4:0];
329
330         fb_addr_x <= writedata[10:5];
331         fb_addr_y <= writedata[4:0];
332         fb_writedata <= writedata[11];
333         fb_WE <= writedata[12];
334     end else begin
335         data_out <= {31'h0, fb_readdata};
336         fb_addr_x <= fbvx_prev;
337         fb_addr_y <= fbvy_prev;
338         fb_WE <= 1'b0;

```

```

338         end
339     end
340
341     //Read/write stack
342     18'h18 : begin
343         $display("READ/WRITE STACK NOT
344             ↳ IMPLEMENTED");
345         data_out <= 32'h18;
346     end
347
348     //MODIFY MEMORY
349     18'h19 : begin
350         if(write) begin
351             memaddr1 <= writedata[19:8];
352             memWE1 <= writedata[20] & write;
353             memwritedata1 <= writedata[7:0];
354
355             mem_addr_prev <= writedata[19:8];
356         end else begin
357             data_out <= {12'h0, mem_addr_prev,
358                         ↳ memreaddata1};
359             memWE1 <= 1'b0;
360             memaddr1 <= mem_addr_prev;
361         end
362     end
363
364     //Load single instruction
365     18'h1A : begin
366         if(write)
367             cpu_instruction <= writedata[15:0];
368         data_out <= {stage[15:0],
369                         ↳ cpu_instruction};
370         stage <= 32'h0;
371     end
372
373     18'h1B : begin
374         //Add initial values for code
375         pc <= 12'h200;

```

```

373
374         memWE1 <= 1'b0;
375         memWE2 <= 1'b0;
376         cpu_instruction <= 16'h0;
377         delay_timer_write_enable <= 1'b0;
378         sound_timer_write_enable <= 1'b0;
379         I <= 16'h0;
380         regWE1 <= 1'b0;
381         regWE2 <= 1'b0;
382
383         fbreset <= 1'b0;
384         fb_addr_y <= 5'b0;
385         fb_addr_x <= 6'b0;
386         fb_writedata <= 1'b0;
387         fb_WE <= 1'b0;
388
389         stack_op <= STACK_HOLD;
390
391         state <= Chip8_PAUSED;
392         cpu_instruction <= 16'h0;
393         stage <= 32'h0;
394
395         stack_reset <= 1'b1;
396
397         fbvx_prev <= 6'h0;
398         fbvy_prev <= 5'h0;
399         mem_addr_prev <= 12'h0;
400
401         sound_on <= 1'b0;
402         chipselect_happened <= 1'b0;
403
404         fb_paused <= 1'b1;
405
406         halt_for_keypress <= 1'b0;
407     end
408
409     default: begin
410         data_out <= 32'd101;

```

```

411         end
412
413     endcase
414
415     end else if(chipselect_happened) begin
416         // sound_timer_write_enable <=
417         //    → chip_sound_timer_write_enable_prev;
418         // delay_timer_write_enable <=
419         //    → chip_delay_timer_write_enable_prev;
420         // reg_addr1 <= chip_reg_addr1_prev;
421         // regWE1 <= chip_regWE1_prev;
422         // reg_writedata1 <= chip_reg_writedata1_prev;
423         // fb_addr_x <= chip_fb_addr_x_prev;
424         // fb_addr_y <= chip_fb_addr_y_prev;
425         // fb_writedata <= chip_fb_writedata_prev;
426         // fb_WE <= chip_fb_WE_prev;
427         // memaddr1 <= chip_memaddr1_prev;
428         // memWE1 <= chip_memWE1_prev;
429         // memwritedata1 <= chip_memwritedata1_prev;
430
431         chipselect_happened <= 1'b0;
432         stack_reset <= 1'b0;
433     end else begin
434         fb_paused <= state == Chip8_PAUSED;
435
436         case (state)
437             Chip8_RUNNING: begin
438                 sound_on <= sound_timer_out;
439
440                 if(halt_for_keypress) begin
441                     if(ispressed) begin
442                         halt_for_keypress <= 1'b0;
443                     end
444                 end else if(stage == 32'h0) begin
445                     memaddr1 <= pc;
446                     memaddr2 <= pc + 12'h1;
447                     cpu_instruction <= 16'h0;
448
449             end

```

```

447     bit_ovewritten <= 1'b0;
448     is_drawing <= 1'b0;
449
450     delay_timer_write_enable <= 1'b0;
451     sound_timer_write_enable <= 1'b0;
452     regWE1 <= 1'b0;
453     regWE2 <= 1'b0;
454     memWE1 <= 1'b0;
455     memWE2 <= 1'b0;
456     stack_op <= STACK_HOLD;
457 end else if (stage == 32'h1) begin
458     memaddr1 <= pc;
459     memaddr2 <= pc + 12'h1;
460     cpu_instruction <= {memreaddata1,
461                         → memreaddata2};
462     last_stage <= stage;
463 end else if (stage >= 32'h2) begin
464     last_stage <= stage;
465     if(cpu_delay_timer_WE) begin
466         delay_timer_write_enable <= 1'b1;
467         delay_timer_data <=
468             → cpu_delay_timer_writedata;
469     end else begin
470         delay_timer_write_enable <= 1'b0;
471     end
472
473     if(cpu_sound_timer_WE) begin
474         sound_timer_write_enable <= 1'b1;
475         sound_timer_data <=
476             → cpu_sound_timer_writedata;
477     end else begin
478         sound_timer_write_enable <= 1'b0;
479     end
480
481     if(cpu_reg_WE1) begin
482         regWE1 <= 1'b1;
483         reg_writedata1 <=
484             → cpu_reg_writedata1;

```

```

481     end else begin
482         regWE1 <= 1'b0;
483     end
484
485     if(cpu_is_drawing) begin
486         is_drawing <= 1'b1;
487     end
488
489     if(cpu_bit_overwritten) begin
490         bit_ovewritten <= 1'b1;
491     end
492
493     if(stage == 32'd30000 && is_drawing)
494         begin
495             regWE2 <= 1'b1;
496             reg_writedata2 <= {7'h0,
497                 → bit_ovewritten};
498             reg_addr2 <= 4'hF; //Setting VF
499                 → register to write
500         end else if(cpu_reg_WE2) begin
501             regWE2 <= 1'b1;
502             reg_writedata2 <=
503                 → cpu_reg_writedata2;
504             reg_addr2 <= cpu_reg_addr2;
505         end else begin
506             regWE2 <= 1'b0;
507             reg_addr2 <= cpu_reg_addr2;
508         end
509
510     if(cpu_reg_I_WE) begin
511         I <= cpu_reg_I_writedata;
512     end
513
514     if(cpu_stk_op == STACK_PUSH) begin
515         stack_op <= STACK_PUSH;
516         stack_writedata <=
517             → cpu_stk_writedata;
518     end

```

```

514
515          //next_pc final modification on
516          //→ NEXT_PC_WRITE_STAGE
517          if((stage >= NEXT_PC_WRITE_STAGE -
518              → 32'h3) & (stage <=
519              → NEXT_PC_WRITE_STAGE)) begin
520              if(cpu_stk_op == STACK_POP) begin
521                  stack_op <= STACK_POP;
522                  next_pc <=
523                      → stack_outdata[11:0];
524              end else begin
525                  case (cpu_pc_src)
526                      PC_SRC_ALU : next_pc <=
527                          → cpu_PC_writedata;
528                      PC_SRC_SKIP : next_pc <=
529                          → pc + 12'd4;
530                      PC_SRC_NEXT : next_pc <=
531                          → pc + 12'd2;
532                      default : next_pc <= pc
533                          → /*default next_pc <=
534                          → pc + 12'd2*/;
535                  endcase
536              end
537          end
538
539          if(cpu_fb_WE) begin
540              fb_writedata <= cpu_fb_writedata;
541              fb_WE <= cpu_fb_WE;
542          end else begin
543              fb_WE <= 1'b0;
544          end
545
546          if(cpu_halt_for_keypress & !ispressed)
547              → begin
548                  halt_for_keypress <= 1'b1;
549              end
550
551          if(cpu_mem_request) begin

```

```

542         memaddr1 <= cpu_mem_addr1;
543         memaddr2 <= cpu_mem_addr2;
544         memwritedata1 <=
545             → cpu_mem_writedata1;
546         memwritedata2 <=
547             → cpu_mem_writedata2;
548         memWE1 <= cpu_mem_WE1;
549         memWE2 <= cpu_mem_WE2;
550     end
551
552     //Always
553     reg_addr1 <= cpu_reg_addr1;
554     fb_addr_x <= cpu_fb_addr_x;
555     fb_addr_y <= cpu_fb_addr_y;
556     // memaddr1 <= cpu_mem_addr1;
557     // memaddr2 <= cpu_mem_addr2;
558 end
559
560 //Cap of 50000, since 1000
561 // instructions/sec is reasonable
562 if(!halt_for_keypress) begin
563     if(stage >= CPU_CYCLE_LENGTH) begin
564         stage <= 32'h0;
565         pc <= next_pc;
566     end
567     else if (stage == 32'h1) begin
568         if(stage == last_stage) stage <=
569             → 32'h2;
570         else stage <= 32'h1;
571     end else if(stage == 32'h2) begin
572         if(stage == last_stage) stage <=
573             → 32'h3;
574         else stage <= 32'h2;
575     end
576
577     else begin
578         stage <= stage + 32'h1;
579     end

```

```

575           end
576       end
577   Chip8_RUN_INSTRUCTION: begin
578       // sound_on <= 1'b1;
579   end
580   Chip8_PAUSED: begin
581       // sound_on <= 1'b1;
582   end
583   default : /* default */;
584 endcase
585 end
586
587
588 Chip8_framebuffer framebuffer(
589     .clk(clk),
590     .reset(fbreset),
591     .fb_addr_y(fb_addr_y),
592     .fb_addr_x(fb_addr_x),
593     .fb_writedata(fb_writedata),
594     .fb_WE(fb_WE),
595     .fb_readdata(fb_readdata),
596     .is_paused (fb_paused),
597     .VGA_R(VGA_R),
598     .VGA_G(VGA_G),
599     .VGA_B(VGA_B),
600     .VGA_CLK(VGA_CLK),
601     .VGA_HS(VGA_HS),
602     .VGA_VS(VGA_VS),
603     .VGA_BLANK_n(VGA_BLANK_n),
604     .VGA_SYNC_n(VGA_SYNC_n)
605 );
606
607 memory memory(
608     .address_a(memaddr1),
609     .address_b(memaddr2),
610     .clock(clk),
611     .data_a(memwritedata1),
612     .data_b(memwritedata2),

```

```

613     .wren_a(memWE1),
614     .wren_b(memWE2),
615     .q_a(memreaddata1),
616     .q_b(memreaddata2)
617 );
618
619 reg_file reg_file(
620     .clock(clk),
621     .address_a(reg_addr1),
622     .address_b(reg_addr2),
623     .data_a(reg_writedata1),
624     .data_b(reg_writedata2),
625     .wren_a(regWE1),
626     .wren_b(regWE2),
627     .q_a(reg_readdata1),
628     .q_b(reg_readdata2)
629 );
630
631 Chip8_CPU cpu(
632     .cpu_clk(clk),
633     .instruction(cpu_instruction),
634     .reg_readdata1(reg_readdata1),
635     .reg_readdata2(reg_readdata2),
636     .mem_readdata1(memreaddata1),
637     .mem_readdata2(memreaddata2),
638     .reg_I_readdata(I),
639     .delay_timer_readdata(delay_timer_output_data),
640     .key_pressed(ispressed),
641     .key_press(key),
642     .PC_readdata(pc),
643     .stage(stage),
644     .top_level_state(state),
645     .delay_timer_WE(cpu_delay_timer_WE),
646     .sound_timer_WE(cpu_sound_timer_WE),
647     .delay_timer_writedata(cpu_delay_timer_writedata),
648     .sound_timer_writedata(cpu_sound_timer_writedata),
649     .pc_src(cpu_pc_src),
650     .PC_writedata(cpu_PC_writedata),

```

```

651     .reg_WE1(cpu_reg_WE1),
652     .reg_WE2(cpu_reg_WE2),
653     .reg_addr1(cpu_reg_addr1),
654     .reg_addr2(cpu_reg_addr2),
655     .reg_writedata1(cpu_reg_writedata1),
656     .reg_writedata2(cpu_reg_writedata2),
657     .mem_WE1(cpu_mem_WE1),
658     .mem_WE2(cpu_mem_WE2),
659     .mem_addr1(cpu_mem_addr1),
660     .mem_addr2(cpu_mem_addr2),
661     .mem_request      (cpu_mem_request),
662     .mem_writedata1(cpu_mem_writedata1),
663     .mem_writedata2(cpu_mem_writedata2),
664     .reg_I_WE(cpu_reg_I_WE),
665     .reg_I_writedata(cpu_reg_I_writedata),
666     .fbreset(cpu_fbreset),
667
668     .fb_addr_y(cpu_fb_addr_y),
669     .fb_addr_x(cpu_fb_addr_x),
670     .fb_writedata(cpu_fb_writedata),
671     .fb_WE(cpu_fb_WE),
672     .fb_readdata(fb_readdata),
673
674     .bit_overwritten(cpu_bit_overwritten),
675     .isDrawing(cpu_is_drawing),
676
677     .stk_reset(cpu_stk_reset),
678     .stk_op(cpu_stk_op),
679     .stk_writedata(cpu_stk_writedata),
680
681     .halt_for_keypress(cpu_halt_for_keypress)
682 );
683
684 Chip8_SoundController sound(
685     .OSC_50_B8A(OSC_50_B8A),
686     .AUD_ADCLRCK(AUD_ADCLRCK),
687     .AUD_ADCDAT(AUD_ADCDAT),
688     .AUD_DACLRCK(AUD_DACLRCK),

```

```

689     .AUD_DACDAT(AUD_DACDAT),
690     .AUD_XCK(AUD_XCK),
691     .AUD_BCLK(AUD_BCLK),
692     .AUD_I2C_SCLK(AUD_I2C_SCLK),
693     .AUD_I2C_SDAT(AUD_I2C_SDAT),
694     .AUD_MUTE(AUD_MUTE),
695
696     .clk(clk),
697     .is_on(sound_on),
698     .reset(sound_reset)
699 );
700
701 clk_div clk_div(
702     .clk_in(clk),
703     .reset(clk_div_reset),
704     .clk_out(clk_div_clk_out)
705 );
706
707 timer delay_timer(
708     .clk(clk),
709     .clk_60(clk_div_clk_out),
710     .write_enable(delay_timer_write_enable),
711     .data(delay_timer_data),
712     .out(delay_timer_out),
713     .output_data (delay_timer_output_data)
714 );
715
716 timer sound_timer(
717     .clk(clk),
718     .clk_60(clk_div_clk_out),
719     .write_enable(sound_timer_write_enable),
720     .data(sound_timer_data),
721     .out(sound_timer_out),
722     .output_data (sound_timer_output_data)
723 );
724
725 Chip8_Stack stack (
726     .reset(stack_reset),

```

```

727     .cpu_clk(clk),
728     .op(stack_op),
729     .writedata(stack_writedata),
730     .outdata(stack_outdata)
731   );
732
733
734 endmodule

```

### 7.1.16 SoCKit\_top.sv

```

1 // 
2 // =====
3 // Copyright (c) 2013 by Terasic Technologies Inc.
4 //
5 // =====
6 // 
7 // Permission:
8 //
9 //   Terasic grants permission to use and modify this code
10 //   for use
11 //   in synthesis for all Terasic Development Boards and
12 //   Altera Development
13 //   Kits made by Terasic. Other use of this code,
14 //   including the selling
15 //   ,duplication, or modification of any portion is
16 //   strictly prohibited.
17 //
18 // Disclaimer:
19 //
20 //   This VHDL/Verilog or C/C++ source code is intended as a
21 //   design reference
22 //   which illustrates how these types of functions can be
23 //   implemented.
24 //   It is the user's responsibility to verify their design
25 //   for
26 //   consistency and functionality through the use of formal

```

```

18 // verification methods. Terasic provides no warranty
19 // regarding the use
20 // or functionality of this code.
21 //
22 //
23 // Terasic Technologies Inc
24 // 9F., No.176, Sec.2, Gongdao 5th Rd, East Dist, Hsinchu
25 // City, 30070. Taiwan
26 //
27 // web: http://www.terasic.com/
28 // email: support@terasic.com
29 //
30 //
31 // Major Functions: SoCKit_Default
32 //
33 //
34 //
35 //
36 // Revision History :
37 //
38 // Ver : / Author : / Mod. Date : / Changes
39 // Made:
40 // V1.0 : / xinxian : / 04/02/13 : / Initial
41 // Revision
42 //define ENABLE_DDR3
43 //define ENABLE_HPS
44 //define ENABLE_HSMC_XCVR
45

```

```

46 module SoCKit_top(
47
48     //////////////AUD///////////
49     AUD_ADCDAT,
50     AUD_ADCLRCK,
51     AUD_BCLK,
52     AUD_DACDAT,
53     AUD_DACLRCK,
54     AUD_I2C_SCLK,
55     AUD_I2C_SDAT,
56     AUD_MUTE,
57     AUD_XCK,
58
59 '#ifdef ENABLE_DDR3
60     //////////////DDR3/////////
61     DDR3_A,
62     DDR3_BA,
63     DDR3_CAS_n,
64     DDR3_CKE,
65     DDR3_CK_n,
66     DDR3_CK_p,
67     DDR3_CS_n,
68     DDR3_DM,
69     DDR3_DQ,
70     DDR3_DQS_n,
71     DDR3_DQS_p,
72     DDR3_ODT,
73     DDR3_RAS_n,
74     DDR3_RESET_n,
75     DDR3_RZQ,
76     DDR3_WE_n,
77 '#endif /*ENABLE_DDR3*/
78
79     //////////////FAN/////////
80     FAN_CTRL,
81
82 '#ifdef ENABLE_HPS
83     //////////////HPS/////////

```

```
84     HPS_CLOCK_25,
85     HPS_CLOCK_50,
86     HPS_CONV_USB_n,
87     HPS_DDR3_A,
88     HPS_DDR3_BA,
89     HPS_DDR3_CAS_n,
90     HPS_DDR3_CKE,
91     HPS_DDR3_CK_n,
92     HPS_DDR3_CK_p,
93     HPS_DDR3_CS_n,
94     HPS_DDR3_DM,
95     HPS_DDR3_DQ,
96     HPS_DDR3_DQS_n,
97     HPS_DDR3_DQS_p,
98     HPS_DDR3_ODT,
99     HPS_DDR3_RAS_n,
100    HPS_DDR3_RESET_n,
101    HPS_DDR3_RZQ,
102    HPS_DDR3_WE_n,
103    HPS_ENET_GTX_CLK,
104    HPS_ENET_INT_n,
105    HPS_ENET_MDC,
106    HPS_ENET_MDIO,
107    HPS_ENET_RESET_n,
108    HPS_ENET_RX_CLK,
109    HPS_ENET_RX_DATA,
110    HPS_ENET_RX_DV,
111    HPS_ENET_TX_DATA,
112    HPS_ENET_TX_EN,
113    HPS_FLASH_DATA,
114    HPS_FLASH_DCLK,
115    HPS_FLASH_NCS0,
116    HPS_GSENSOR_INT,
117    HPS_I2C_CLK,
118    HPS_I2C_SDA,
119    HPS_KEY,
120    HPS_LCM_D_C,
121    HPS_LCM_RST_N,
```

```

122     HPS_LCM_SPIM_CLK,
123     HPS_LCM_SPIM_MISO,
124     HPS_LCM_SPIM_MOSI,
125     HPS_LCM_SPIM_SS,
126     HPS_LED,
127     HPS_LTC_GPIO,
128     HPS_RESET_n,
129     HPS_SD_CLK,
130     HPS_SD_CMD,
131     HPS_SD_DATA,
132     HPS_SPIM_CLK,
133     HPS_SPIM_MISO,
134     HPS_SPIM_MOSI,
135     HPS_SPIM_SS,
136     HPS_SW,
137     HPS_UART_RX,
138     HPS_UART_TX,
139     HPS_USB_CLKOUT,
140     HPS_USB_DATA,
141     HPS_USB_DIR,
142     HPS_USB_NXT,
143     HPS_USB_RESET_PHY,
144     HPS_USB_STP,
145     HPS_WARM_RST_n,
146 'endif /*ENABLE_HPS*/
147
148 //////////////HSMC///////////
149 HSMC_CLKIN_n,
150 HSMC_CLKIN_p,
151 HSMC_CLKOUT_n,
152 HSMC_CLKOUT_p,
153 HSMC_CLK_INO,
154 HSMC_CLK_OUT0,
155 HSMC_D,
156
157 'ifdef ENABLE_HSMC_XCVR
158
159     HSMC_GXB_RX_p,

```

```

160      HSMC_GXB_TX_p,
161      HSMC_REF_CLK_p,
162  'endif
163      HSMC_RX_n,
164      HSMC_RX_p,
165      HSMC_SCL,
166      HSMC_SDA,
167      HSMC_TX_n,
168      HSMC_TX_p,
169
170      //////////////IRDA///////////
171      IRDA_RXD,
172
173      //////////////KEY///////////
174      KEY,
175
176      //////////////LED///////////
177      LED,
178
179      //////////////OSC///////////
180      OSC_50_B3B,
181      OSC_50_B4A,
182      OSC_50_B5B,
183      OSC_50_B8A,
184
185      //////////////PCIE///////////
186      PCIE_PERST_n,
187      PCIE_WAKE_n,
188
189      //////////////RESET/////////
190      RESET_n,
191
192      //////////////SI5338/////////
193      SI5338_SCL,
194      SI5338_SDA,
195
196      //////////////SW///////////
197      SW,

```

```

198
199      //////////////TEMP///////////
200      TEMP_CS_n,
201      TEMP_DIN,
202      TEMP_DOUT,
203      TEMP_SCLK,
204
205      //////////////USB///////////
206      USB_B2_CLK,
207      USB_B2_DATA,
208      USB_EMPTY,
209      USB_FULL,
210      USB_OE_n,
211      USB_RD_n,
212      USB_RESET_n,
213      USB_SCL,
214      USB_SDA,
215      USB_WR_n,
216
217      //////////////VGA///////////
218      VGA_B,
219      VGA_BLANK_n,
220      VGA_CLK,
221      VGA_G,
222      VGA_HS,
223      VGA_R,
224      VGA_SYNC_n,
225      VGA_VS,
226      //////////////hps///////////
227      memory_mem_a,
228      memory_mem_ba,
229      memory_mem_ck,
230      memory_mem_ck_n,
231      memory_mem_cke,
232      memory_mem_cs_n,
233      memory_mem_ras_n,
234      memory_mem_cas_n,
235      memory_mem_we_n,

```

```

236     memory_mem_reset_n,
237     memory_mem_dq,
238     memory_mem_dqs,
239     memory_mem_dqs_n,
240     memory_mem_odt,
241     memory_mem_dm,
242     memory_oct_rzqin,
243     hps_io_hps_io_emac1_inst_TX_CLK,
244     hps_io_hps_io_emac1_inst_RXD0,
245     hps_io_hps_io_emac1_inst_RXD1,
246     hps_io_hps_io_emac1_inst_RXD2,
247     hps_io_hps_io_emac1_inst_RXD3,
248     hps_io_hps_io_emac1_inst_RXD0,
249     hps_io_hps_io_emac1_inst_MDIO,
250     hps_io_hps_io_emac1_inst_MDC,
251     hps_io_hps_io_emac1_inst_RX_CTL,
252     hps_io_hps_io_emac1_inst_TX_CTL,
253     hps_io_hps_io_emac1_inst_RX_CLK,
254     hps_io_hps_io_emac1_inst_RXD1,
255     hps_io_hps_io_emac1_inst_RXD2,
256     hps_io_hps_io_emac1_inst_RXD3,
257     hps_io_hps_io_qspi_inst_I00,
258     hps_io_hps_io_qspi_inst_I01,
259     hps_io_hps_io_qspi_inst_I02,
260     hps_io_hps_io_qspi_inst_I03,
261     hps_io_hps_io_qspi_inst_SS0,
262     hps_io_hps_io_qspi_inst_CLK,
263     hps_io_hps_io_sdio_inst_CMD,
264     hps_io_hps_io_sdio_inst_D0,
265     hps_io_hps_io_sdio_inst_D1,
266     hps_io_hps_io_sdio_inst_CLK,
267     hps_io_hps_io_sdio_inst_D2,
268     hps_io_hps_io_sdio_inst_D3,
269     hps_io_hps_io_usb1_inst_D0,
270     hps_io_hps_io_usb1_inst_D1,
271     hps_io_hps_io_usb1_inst_D2,
272     hps_io_hps_io_usb1_inst_D3,
273     hps_io_hps_io_usb1_inst_D4,
```

```

274     hps_io_hps_io_usb1_inst_D5,
275     hps_io_hps_io_usb1_inst_D6,
276     hps_io_hps_io_usb1_inst_D7,
277     hps_io_hps_io_usb1_inst_CLK,
278     hps_io_hps_io_usb1_inst_STP,
279     hps_io_hps_io_usb1_inst_DIR,
280     hps_io_hps_io_usb1_inst_NXT,
281     hps_io_hps_io_spim0_inst_CLK,
282     hps_io_hps_io_spim0_inst_MOSI,
283     hps_io_hps_io_spim0_inst_MISO,
284     hps_io_hps_io_spim0_inst_SS0,
285     hps_io_hps_io_spim1_inst_CLK,
286     hps_io_hps_io_spim1_inst_MOSI,
287     hps_io_hps_io_spim1_inst_MISO,
288     hps_io_hps_io_spim1_inst_SS0,
289     hps_io_hps_io_uart0_inst_RX,
290     hps_io_hps_io_uart0_inst_TX,
291     hps_io_hps_io_i2c1_inst_SDA,
292     hps_io_hps_io_i2c1_inst_SCL,
293     hps_io_hps_io_gpio_inst_GPIO000
294 );
295
296 //=====
297 // PORT declarations
298 //=====
299
300 ////////// AUD //////////
301 input
302   ↳ AUD_ADCDAT;
303 inout
304   ↳ AUD_ADCLRCK;
305 inout
306   ↳ AUD_BCLK;
307 output
308   ↳ AUD_DACDAT;
309 inout
310   ↳ AUD_DACLRCK;

```

```

306     output
307         ↳ AUD_I2C_SCLK;
308     inout
309         ↳ AUD_I2C_SDAT;
310     output
311         ↳ AUD_MUTE;
312     output
313         ↳ AUD_XCK;
314
315     'ifdef ENABLE_DDR3
316         ////////////// DDR3 ///////////
317     output [14:0]           DDR3_A;
318     output [2:0]           DDR3_BA;
319     output
320         ↳ DDR3_CAS_n;
321     output
322         ↳ DDR3_CKE;
323     output
324         ↳ DDR3_CK_n;
325     output
326         ↳ DDR3_CK_p;
327     output
328         ↳ DDR3_CS_n;
329     output [3:0]           DDR3_DM;
330     inout [31:0]
331     inout [3:0]           DDR3_DQ;
332     inout [3:0]
333     inout [3:0]           DDR3_DQS_n;
334     output
335         ↳ DDR3_DQS_p;
336     output
337         ↳ DDR3_ODT;
338     output
339         ↳ DDR3_RAS_n;
340     output
341         ↳ DDR3_RESET_n;
342     input
343         ↳ DDR3_RZQ;
344     output
345         ↳ DDR3_WE_n;
346
347     'endif /*ENABLE_DDR3*/
348
349

```

```

331      ////////////// FAN ///////////
332      output
333          ↳  FAN_CTRL;
334
335      'ifdef ENABLE_HPS
336          ////////////// HPS ///////////
337          input
338              ↳  HPS_CLOCK_25;
339          input
340              ↳  HPS_CLOCK_50;
341          input
342              ↳  HPS_CONV_USB_n;
343          output [14:0]           HPS_DDR3_A;
344          output [2:0]            HPS_DDR3_BA;
345          output
346              ↳  HPS_DDR3_CAS_n;
347          output
348              ↳  HPS_DDR3_CKE;
349          output
350              ↳  HPS_DDR3_CK_n;
351          output
352              ↳  HPS_DDR3_CK_p;
353          output
354              ↳  HPS_DDR3_CS_n;
355          output [3:0]           HPS_DDR3_DM;
356          inout [31:0]           HPS_DDR3_DQ;
357          inout [3:0]            HPS_DDR3_DQS_n;
358          inout [3:0]            HPS_DDR3_DQS_p;
359          output
360              ↳  HPS_DDR3_ODT;
361          output
362              ↳  HPS_DDR3_RAS_n;
363          output
364              ↳  HPS_DDR3_RESET_n;
365          input
366              ↳  HPS_DDR3_RZQ;
367          output
368              ↳  HPS_DDR3_WE_n;

```

```

355     input
356         ↳ HPS_ENET_GTX_CLK;
357     input
358         ↳ HPS_ENET_INT_n;
359     output
360         ↳ HPS_ENET_MDC;
361     inout
362         ↳ HPS_ENET_MDIO;
363     output
364         ↳ HPS_ENET_RESET_n;
365     input
366         ↳ HPS_ENET_RX_CLK;
367     input [3:0]                                HPS_ENET_RX_DATA;
368     input
369         ↳ HPS_ENET_RX_DV;
370     output [3:0]                                HPS_ENET_TX_DATA;
371     output
372         ↳ HPS_ENET_TX_EN;
373     inout [3:0]                                HPS_FLASH_DATA;
374     output
375         ↳ HPS_FLASH_DCLK;
376     output
377         ↳ HPS_FLASH_NCS0;
378     input
379         ↳ HPS_GSENSOR_INT;
380     inout
381         ↳ HPS_I2C_CLK;
382     inout
383         ↳ HPS_I2C_SDA;
384     inout [3:0]                                HPS_KEY;
385     output
386         ↳ HPS_LCM_D_C;
387     output
388         ↳ HPS_LCM_RST_N;
389     input
390         ↳ HPS_LCM_SPIM_CLK;
391     inout
392         ↳ HPS_LCM_SPIM_MISO;

```

```

376   output
377     ↳ HPS_LCM_SPIM_MOSI;
378   output
379     ↳ HPS_LCM_SPIM_SS;
380   output [3:0]                                HPS_LED;
381   inout
382     ↳ HPS_LTC_GPIO;
383   input
384     ↳ HPS_RESET_n;
385   output
386     ↳ HPS_SD_CLK;
387   inout
388     ↳ HPS_SD_CMD;
389   inout [3:0]                                HPS_SD_DATA;
390   output
391     ↳ HPS_SPIM_CLK;
392   input
393     ↳ HPS_SPIM_MISO;
394   output
395     ↳ HPS_SPIM_MOSI;
396   output
397     ↳ HPS_SPIM_SS;
398   input [3:0]                                HPS_SW;
399   input
400     ↳ HPS_UART_RX;
401   output
402     ↳ HPS_UART_TX;
403   input
404     ↳ HPS_USB_CLKOUT;
405   inout [7:0]                                HPS_USB_DATA;
406   input
407     ↳ HPS_USB_DIR;
408   input
409     ↳ HPS_USB_NXT;
410   output
411     ↳ HPS_USB_RESET_PHY;
412   output
413     ↳ HPS_USB_STP;

```

```

397     input
398         ↳ HPS_WARM_RST_n;
399 'endif /*ENABLE_HPS*/
400
401 ////////////// HSMC //////////
402     input [2:1]                                HSMC_CLKIN_n;
403     input [2:1]                                HSMC_CLKIN_p;
404     output [2:1]                               HSMC_CLKOUT_n;
405     output [2:1]                               HSMC_CLKOUT_p;
406     input
407         ↳ HSMC_CLK_IN0;
408     output
409         ↳ HSMC_CLK_OUT0;
410     inout [3:0]                                HSMC_D;
411
412 'ifdef ENABLE_HSMC_XCVR
413     input [7:0]                                HSMC_GXB_RX_p;
414     output [7:0]                               HSMC_GXB_TX_p;
415     input
416         ↳ HSMC_REF_CLK_p;
417 'endif
418     inout [16:0]                               HSMC_RX_n;
419     inout [16:0]                               HSMC_RX_p;
420     output
421         ↳ HSMC_SCL;
422     inout
423         ↳ HSMC_SDA;
424     inout [16:0]                               HSMC_TX_n;
425     inout [16:0]                               HSMC_TX_p;
426
427 ////////////// IRDA //////////
428     input
429         ↳ IRDA_RXD;
430
431 ////////////// KEY //////////
432     input [3:0]                                KEY;
433
434 ////////////// LED //////////
435     output [3:0]                               LED;

```

```

428
429   ////////////// OSC ///////////
430   input
431     ↳ OSC_50_B3B;
432   input
433     ↳ OSC_50_B4A;
434   input
435     ↳ OSC_50_B5B;
436   input
437     ↳ OSC_50_B8A;

438
439   ////////////// PCIE ///////////
440   input
441     ↳ PCIE_PERST_n;
442   input
443     ↳ PCIE_WAKE_n;

444
445   ////////////// RESET ///////////
446   input
447     ↳ RESET_n;

448
449   ////////////// SI5338 ///////////
450   inout
451     ↳ SI5338_SCL;
452   inout
453     ↳ SI5338_SDA;

454
455   ////////////// SW ///////////
456   input [3:0]
457     ↳ SW;

458
459   ////////////// TEMP ///////////
460   output
461     ↳ TEMP_CS_n;
462   output
463     ↳ TEMP_DIN;
464   input
465     ↳ TEMP_DOUT;
466   output
467     ↳ TEMP_SCLK;

```

```

454
455 ////////////// USB ///////////
456
457 input
458    ↳  USB_B2_CLK;
459 inout [7:0]                               USB_B2_DATA;
460 output
461    ↳  USB_EMPTY;
462 output
463    ↳  USB_FULL;
464
465 input
466    ↳  USB_OE_n;
467 input
468    ↳  USB_RD_n;
469 input
470    ↳  USB_RESET_n;
471
472 inout                                     USB_SCL;
473 inout                                     USB_SDA;
474
475 input
476    ↳  USB_WR_n;
477
478 ////////////// VGA ///////////
479
480 output [7:0]                             VGA_B;
481 output
482    ↳  VGA_BLANK_n;
483 output                                     VGA_CLK;
484
485 output [7:0]                             VGA_G;
486 output                                     VGA_HS;
487
488 output [7:0]                             VGA_R;
489 output
490    ↳  VGA_SYNC_n;
491 output                                     VGA_VS;
492
493 //////////////hps pin/////////
494
495 output wire [14:0]                         memory_mem_a;
496 output wire [2:0]                           memory_mem_ba;
497 output wire                               memory_mem_ck;
498 output wire                               memory_mem_ck_n;
499 output wire                               memory_mem_cke;
500

```

```

483   output wire                               memory_mem_cs_n;
484   output wire                               memory_mem_ras_n;
485   output wire                               memory_mem_cas_n;
486   output wire                               memory_mem_we_n;
487   output wire                               memory_mem_reset_n;
488   inout  wire [31:0]                         memory_mem_dq;
489   inout  wire [3:0]                          memory_mem_dqs;
490   inout  wire [3:0]                          memory_mem_dqs_n;
491   output wire                               memory_mem_odt;
492   output wire [3:0]                         memory_mem_dm;
493   input  wire                                memory_oct_rzqin;
494   output wire
495     ↳ hps_io_hps_io_emac1_inst_TX_CLK;
496   output wire
497     ↳ hps_io_hps_io_emac1_inst_RXD0;
498   output wire
499     ↳ hps_io_hps_io_emac1_inst_RXD1;
500   output wire
501     ↳ hps_io_hps_io_emac1_inst_RXD2;
502   output wire
503     ↳ hps_io_hps_io_emac1_inst_RXD3;
504   input  wire                                hps_io_hps_io_emac1_inst_RXD0;
505   inout  wire
506     ↳ hps_io_hps_io_emac1_inst_MDIO;
507   output wire
508     ↳ hps_io_hps_io_emac1_inst_MDC;
509   input  wire
510     ↳ hps_io_hps_io_emac1_inst_RX_CTL;
511   output wire
512     ↳ hps_io_hps_io_emac1_inst_TX_CTL;
513   input  wire
514     ↳ hps_io_hps_io_emac1_inst_RX_CLK;
515   input  wire
516     ↳ hps_io_hps_io_emac1_inst_RXD1;
517   input  wire
518     ↳ hps_io_hps_io_emac1_inst_RXD2;

```

```

507   input wire
508     ↳ hps_io_hps_io_emac1_inst_RXD3;
509   inout wire
510     ↳ hps_io_hps_io_qspi_inst_I00;
511   inout wire
512     ↳ hps_io_hps_io_qspi_inst_I01;
513   inout wire
514     ↳ hps_io_hps_io_qspi_inst_I02;
515   inout wire
516     ↳ hps_io_hps_io_qspi_inst_I03;
517   output wire
518     ↳ hps_io_hps_io_qspi_inst_SS0;
519   output wire
520     ↳ hps_io_hps_io_qspi_inst_CLK;
521   inout wire
522     ↳ hps_io_hps_io_sdio_inst_CMD;
523   inout wire
524     ↳ hps_io_hps_io_sdio_inst_D0;
525   inout wire
      ↳ hps_io_hps_io_sdio_inst_D1;
526   output wire
527     ↳ hps_io_hps_io_sdio_inst_CLK;
528   inout wire
529     ↳ hps_io_hps_io_sdio_inst_D2;
530   inout wire
531     ↳ hps_io_hps_io_sdio_inst_D3;
532   inout wire
533     ↳ hps_io_hps_io_usb1_inst_D0;
534   inout wire
535     ↳ hps_io_hps_io_usb1_inst_D1;
536   inout wire
537     ↳ hps_io_hps_io_usb1_inst_D2;
538   inout wire
539     ↳ hps_io_hps_io_usb1_inst_D3;
540   inout wire
541     ↳ hps_io_hps_io_usb1_inst_D4;
542   inout wire
543     ↳ hps_io_hps_io_usb1_inst_D5;

```

```

526   inout wire
527     ↳ hps_io_hps_io_usb1_inst_D6;
528   inout wire
529     ↳ hps_io_hps_io_usb1_inst_D7;
530   input wire
531     ↳ hps_io_hps_io_usb1_inst_CLK;
532   output wire
533     ↳ hps_io_hps_io_usb1_inst_STP;
534   input wire
535     ↳ hps_io_hps_io_usb1_inst_DIR;
536   input wire
537     ↳ hps_io_hps_io_usb1_inst_NXT;
538   output wire
539     ↳ hps_io_hps_io_spim0_inst_CLK;
540   output wire
541     ↳ hps_io_hps_io_spim0_inst_MOSI;
542   input wire
543     ↳ hps_io_hps_io_spim0_inst_MISO;
544   output wire
545     ↳ hps_io_hps_io_spim0_inst_SS0;
546   output wire
547     ↳ hps_io_hps_io_spim1_inst_CLK;
548   output wire
549     ↳ hps_io_hps_io_spim1_inst_MOSI;
550   input wire
551     ↳ hps_io_hps_io_spim1_inst_MISO;
552   output wire
553     ↳ hps_io_hps_io_spim1_inst_SS0;
554   input wire
555     ↳ hps_io_hps_io_uart0_inst_RX;
556   output wire
557     ↳ hps_io_hps_io_uart0_inst_TX;
558   inout wire
559     ↳ hps_io_hps_io_i2c1_inst_SDA;
560   inout wire
561     ↳ hps_io_hps_io_i2c1_inst_SCL;
562   inout wire
563     ↳ hps_io_hps_io_gpio_inst_GPIO00;

```

```

545 //=====
546 // REG/WIRE declarations
547 //=====
548
549 // For Audio CODEC
550 wire AUD_CTRL_CLK; // For
551   → Audio Controller
552
553 reg [31:0] Cont;
554 wire VGA_CTRL_CLK;
555 wire [9:0] mVGA_R;
556 wire [9:0] mVGA_G;
557 wire [9:0] mVGA_B;
558 wire [19:0] mVGA_ADDR;
559   wire DLY_RST;
560
561 // For VGA Controller
562 wire mVGA_CLK;
563 wire [9:0] mRed;
564 wire [9:0] mGreen;
565 wire [9:0] mBlue;
566   wire VGA_Read; // VGA data
567   → request
568
569 wire [9:0] recon_VGA_R;
570 wire [9:0] recon_VGA_G;
571 wire [9:0] recon_VGA_B;
572
573 // For Down Sample
574 wire [3:0] Remain;
575 wire [9:0] Quotient;
576
577 wire AUD_MUTE;
578
579 // Drive the LEDs with the switches
580 assign LED = SW;
581
582 // Make the FPGA reset cause an HPS reset

```

```

581   reg [19:0]           hps_reset_counter =
582   ↵    20'h0;
583   reg                  hps_fpga_reset_n = 0;
584
585   always @ (posedge OSC_50_B4A) begin
586     if (hps_reset_counter == 20'h ffffff) hps_fpga_reset_n
587     ↵    <= 1;
588     hps_reset_counter <= hps_reset_counter + 1;
589   end
590
591   Chip8 u0 (
592     .clk_clk
593     ↵    (OSC_50_B4A),           // 
594     ↵    clk.clk
595     .reset_reset_n
596     ↵    (hps_fpga_reset_n),    // 
597     ↵    reset.reset_n
598     .memory_mem_a
599     ↵    (memory_mem_a),        // 
600     ↵    memory.mem_a
601     .memory_mem_ba
602     ↵    (memory_mem_ba),       // 
603     ↵    .mem_ba
604     .memory_mem_ck
605     ↵    (memory_mem_ck),       // 
606     ↵    .mem_ck
607     .memory_mem_ck_n
608     ↵    (memory_mem_ck_n),     // 
609     ↵    .mem_ck_n
610     .memory_mem_cke
611     ↵    (memory_mem_cke),      // 
612     ↵    .mem_cke
613     .memory_mem_cs_n
614     ↵    (memory_mem_cs_n),     // 
615     ↵    .mem_cs_n
616     .memory_mem_ras_n
617     ↵    (memory_mem_ras_n),     // 
618     ↵    .mem_ras_n

```

```

599      .memory_mem_cas_n
       ↳  (memory_mem_cas_n),           // 
       ↳  .mem_cas_n
600      .memory_mem_we_n
       ↳  (memory_mem_we_n),           // 
       ↳  .mem_we_n
601      .memory_mem_reset_n
       ↳  (memory_mem_reset_n),         // 
       ↳  .mem_reset_n
602      .memory_mem_dq
       ↳  (memory_mem_dq),             // 
       ↳  .mem_dq
603      .memory_mem_dqs
       ↳  (memory_mem_dqs),             // 
       ↳  .mem_dqs
604      .memory_mem_dqs_n
       ↳  (memory_mem_dqs_n),           // 
       ↳  .mem_dqs_n
605      .memory_mem_odt
       ↳  (memory_mem_odt),             // 
       ↳  .mem_odt
606      .memory_mem_dm
       ↳  (memory_mem_dm),             // 
       ↳  .mem_dm
607      .memory_oct_rzqin
       ↳  (memory_oct_rzqin),           // 
       ↳  .oct_rzqin
608      .hps_io_hps_io_emac1_inst_TX_CLK
       ↳  (hps_io_hps_io_emac1_inst_TX_CLK), // 
       ↳ 
       ↳  .hps_0_hps_io.hps_io_emac1_inst_TX_CLK
609      .hps_io_hps_io_emac1_inst_TXD0
       ↳  (hps_io_hps_io_emac1_inst_TXD0),   // 
       ↳  .hps_io_emac1_inst_TXD0
610      .hps_io_hps_io_emac1_inst_TXD1
       ↳  (hps_io_hps_io_emac1_inst_TXD1),   // 
       ↳  .hps_io_emac1_inst_TXD1

```

```

611   .hps_io_hps_io_emac1_inst_RXD2
612     ↳ (hps_io_hps_io_emac1_inst_RXD2),    //
613       ↳ .hps_io_emac1_inst_RXD2
614   .hps_io_hps_io_emac1_inst_RXD3
615     ↳ (hps_io_hps_io_emac1_inst_RXD3),    //
616       ↳ .hps_io_emac1_inst_RXD3
617   .hps_io_hps_io_emac1_inst_RXD0
618     ↳ (hps_io_hps_io_emac1_inst_RXD0),    //
619       ↳ .hps_io_emac1_inst_RXD0
620   .hps_io_hps_io_emac1_inst_RXD1
621     ↳ (hps_io_hps_io_emac1_inst_RXD1),    //
622       ↳ .hps_io_emac1_inst_RXD1
623   .hps_io_hps_io_emac1_inst_RXD2
624     ↳ (hps_io_hps_io_emac1_inst_RXD2),    //
625       ↳ .hps_io_emac1_inst_RXD2
626   .hps_io_hps_io_emac1_inst_RXD3
627     ↳ (hps_io_hps_io_emac1_inst_RXD3),    //
628       ↳ .hps_io_emac1_inst_RXD3
629   .hps_io_hps_io_qspi_inst_I00
630     ↳ (hps_io_hps_io_qspi_inst_I00),      //
631       ↳ .hps_io_qspi_inst_I00

```

```

623   .hps_io_hps_io_qspi_inst_I01
       ↳ (hps_io_hps_io_qspi_inst_I01),      //
       ↳ .hps_io_qspi_inst_I01
624   .hps_io_hps_io_qspi_inst_I02
       ↳ (hps_io_hps_io_qspi_inst_I02),      //
       ↳ .hps_io_qspi_inst_I02
625   .hps_io_hps_io_qspi_inst_I03
       ↳ (hps_io_hps_io_qspi_inst_I03),      //
       ↳ .hps_io_qspi_inst_I03
626   .hps_io_hps_io_qspi_inst_SS0
       ↳ (hps_io_hps_io_qspi_inst_SS0),      //
       ↳ .hps_io_qspi_inst_SS0
627   .hps_io_hps_io_qspi_inst_CLK
       ↳ (hps_io_hps_io_qspi_inst_CLK),      //
       ↳ .hps_io_qspi_inst_CLK
628   .hps_io_hps_io_sdio_inst_CMD
       ↳ (hps_io_hps_io_sdio_inst_CMD),      //
       ↳ .hps_io_sdio_inst_CMD
629   .hps_io_hps_io_sdio_inst_D0
       ↳ (hps_io_hps_io_sdio_inst_D0),      //
       ↳ .hps_io_sdio_inst_D0
630   .hps_io_hps_io_sdio_inst_D1
       ↳ (hps_io_hps_io_sdio_inst_D1),      //
       ↳ .hps_io_sdio_inst_D1
631   .hps_io_hps_io_sdio_inst_CLK
       ↳ (hps_io_hps_io_sdio_inst_CLK),      //
       ↳ .hps_io_sdio_inst_CLK
632   .hps_io_hps_io_sdio_inst_D2
       ↳ (hps_io_hps_io_sdio_inst_D2),      //
       ↳ .hps_io_sdio_inst_D2
633   .hps_io_hps_io_sdio_inst_D3
       ↳ (hps_io_hps_io_sdio_inst_D3),      //
       ↳ .hps_io_sdio_inst_D3
634   .hps_io_hps_io_usb1_inst_D0
       ↳ (hps_io_hps_io_usb1_inst_D0),      //
       ↳ .hps_io_usb1_inst_D0

```

```

635      .hps_io_hps_io_usb1_inst_D1
635      ↳  (hps_io_hps_io_usb1_inst_D1),      //
635      ↳  .hps_io_usb1_inst_D1
636      .hps_io_hps_io_usb1_inst_D2
636      ↳  (hps_io_hps_io_usb1_inst_D2),      //
636      ↳  .hps_io_usb1_inst_D2
637      .hps_io_hps_io_usb1_inst_D3
637      ↳  (hps_io_hps_io_usb1_inst_D3),      //
637      ↳  .hps_io_usb1_inst_D3
638      .hps_io_hps_io_usb1_inst_D4
638      ↳  (hps_io_hps_io_usb1_inst_D4),      //
638      ↳  .hps_io_usb1_inst_D4
639      .hps_io_hps_io_usb1_inst_D5
639      ↳  (hps_io_hps_io_usb1_inst_D5),      //
639      ↳  .hps_io_usb1_inst_D5
640      .hps_io_hps_io_usb1_inst_D6
640      ↳  (hps_io_hps_io_usb1_inst_D6),      //
640      ↳  .hps_io_usb1_inst_D6
641      .hps_io_hps_io_usb1_inst_D7
641      ↳  (hps_io_hps_io_usb1_inst_D7),      //
641      ↳  .hps_io_usb1_inst_D7
642      .hps_io_hps_io_usb1_inst_CLK
642      ↳  (hps_io_hps_io_usb1_inst_CLK),      //
642      ↳  .hps_io_usb1_inst_CLK
643      .hps_io_hps_io_usb1_inst_STP
643      ↳  (hps_io_hps_io_usb1_inst_STP),      //
643      ↳  .hps_io_usb1_inst_STP
644      .hps_io_hps_io_usb1_inst_DIR
644      ↳  (hps_io_hps_io_usb1_inst_DIR),      //
644      ↳  .hps_io_usb1_inst_DIR
645      .hps_io_hps_io_usb1_inst_NXT
645      ↳  (hps_io_hps_io_usb1_inst_NXT),      //
645      ↳  .hps_io_usb1_inst_NXT
646      .hps_io_hps_io_spim0_inst_CLK
646      ↳  (hps_io_hps_io_spim0_inst_CLK),      //
646      ↳  .hps_io_spim0_inst_CLK

```

```

647   .hps_io_hps_io_spim0_inst_MOSI
648     ↳ (hps_io_hps_io_spim0_inst_MOSI),    //
649       .hps_io_spim0_inst_MOSI
650   .hps_io_hps_io_spim0_inst_MISO
651     ↳ (hps_io_hps_io_spim0_inst_MISO),    //
652       .hps_io_spim0_inst_MISO
653   .hps_io_hps_io_spim0_inst_SS0
654     ↳ (hps_io_hps_io_spim0_inst_SS0),    //
655       .hps_io_spim0_inst_SS0
656   .hps_io_hps_io_spim1_inst_CLK
657     ↳ (hps_io_hps_io_spim1_inst_CLK),    //
658       .hps_io_spim1_inst_CLK
659   .hps_io_hps_io_spim1_inst_MOSI
660     ↳ (hps_io_hps_io_spim1_inst_MOSI),    //
661       .hps_io_spim1_inst_MOSI
662   .hps_io_hps_io_spim1_inst_MISO
663     ↳ (hps_io_hps_io_spim1_inst_MISO),    //
664       .hps_io_spim1_inst_MISO
665   .hps_io_hps_io_spim1_inst_SS0
666     ↳ (hps_io_hps_io_spim1_inst_SS0),    //
667       .hps_io_spim1_inst_SS0
668   .hps_io_hps_io_uart0_inst_RX
669     ↳ (hps_io_hps_io_uart0_inst_RX),    //
670       .hps_io_uart0_inst_RX
671   .hps_io_hps_io_uart0_inst_TX
672     ↳ (hps_io_hps_io_uart0_inst_TX),    //
673       .hps_io_uart0_inst_TX
674   .hps_io_hps_io_i2c1_inst_SDA
675     ↳ (hps_io_hps_io_i2c1_inst_SDA),    //
676       .hps_io_i2c1_inst_SDA
677   .hps_io_hps_io_i2c1_inst_SCL
678     ↳ (hps_io_hps_io_i2c1_inst_SCL),    //
679       .hps_io_i2c1_inst_SCL
680   .chip8_device_VGA_R
681     ↳ (VGA_R),
682   .chip8_device_VGA_G
683     ↳ (VGA_G),

```

```

660      .chip8_device_VGA_B
661          ↳ (VGA_B),
662      .chip8_device_VGA_CLK
663          ↳ (VGA_CLK),
664      .chip8_device_VGA_HS
665          ↳ (VGA_HS),
666      .chip8_device_VGA_VS
667          ↳ (VGA_VS),
668      .chip8_device_VGA_BLANK_n
669          ↳ (VGA_BLANK_n),
670      .chip8_device_OSC_50_B8A
671          ↳ (OSC_50_B8A),
672      .chip8_device_AUD_ADCDAT
673          ↳ (AUD_ADCDAT),
674      .chip8_device_AUD_DACLRCK
675          ↳ (AUD_DACLRCK),
676      .chip8_device_AUD_DACDAT
677          ↳ (AUD_DACDAT),
678      .chip8_device_AUD_XCK
679          ↳ (AUD_XCK),
680      .chip8_device_AUD_BCLK
681          ↳ (AUD_BCLK),
682      .chip8_device_AUD_I2C_SCLK
683          ↳ (AUD_I2C_SCLK),
684      .chip8_device_AUD_I2C_SDAT
685          ↳ (AUD_I2C_SDAT),
686      .chip8_device_AUD_MUTE
687          ↳ (AUD_MUTE),
688      .chip8_device_VGA_SYNC_n
689          ↳ (VGA_SYNC_n)
690      );
691
692 endmodule

```

### 7.1.17 utils.svh

```
1  'ifndef CHIP8_UTILS_SVH
2  'define CHIP8_UTILS_SVH
3
4  function reg inbetween(input [17:0] low, value, high);
5  begin
6      inbetween = value >= low && value <= high;
7  end
8  endfunction
9
10 'endif
```

### 7.1.18 enums.svh

```
1  ****
2  * enums.svh
3  *
4  * Defines the enums used by Chip8_Top, Chip8_ALU, Chip8_CPU
5  *
6  * AUTHORS: David Watkins
7  * Updated: Gabrielle Taylor 5/3/2016
8  * Dependencies:
9
10 ****
11 'ifndef CHIP8_ENUMS
12 'define CHIP8_ENUMS
13
14 /**
15 * ALU_f is an input into the ALU to specify which operation
16 * to execute
17 *
18 * - ALU_f_OR          : bitwise OR
19 * - ALU_f_AND         : bitwise AND
20 * - ALU_f_XOR         : bitwise XOR
21 * - ALU_f_ADD         : Addition
22 * - ALU_f_MINUS       : Subtract
```

```

22     *      - ALU_f_LSHIFT      : Shift left
23     *      - ALU_f_RSHIFT      : Shift right
24     *      - ALU_f_EQUALS      : Equals compare
25     *      - ALU_f_GREATER      : Greater than compare
26     *      - ALU_f_INC          : Increment
27 */
28 typedef enum {
29     ALU_f_OR,
30     ALU_f_AND,
31     ALU_f_XOR,
32     ALU_f_ADD,
33     ALU_f_MINUS,
34     ALU_f_LSHIFT,
35     ALU_f_RSHIFT,
36     ALU_f_EQUALS,
37     ALU_f_GREATER,
38     ALU_f_INC,
39     ALU_f_NOP
40 } ALU_f ;
41
42 /**
43 * PC_SRC defines the behavior of the program counter from
44 * output from the CPU
45 *
46 * PC_SRC_STACK : Read from the current stack pointer
47 * PC_SRC_ALU   : Read the output from the processor
48 * PC_SRC_DEVICE: Read from linux input
49 * PC_SRC_SKIP  : Assign PC = PC + 4
50 * PC_SRC_HOLD  : Assign PC = PC
51 * PC_SRC_NEXT  : Assign PC = PC + 2
52 */
53 typedef enum {
54     PC_SRC_STACK,
55     PC_SRC_ALU,
56     PC_SRC_DEVICE,
57     PC_SRC_SKIP,
58     PC_SRC_HOLD,
59     PC_SRC_NEXT

```

```

59 } PC_SRC;
60
61 /**
62 * Chip8_STATE defines the current state of the emulator
63 *
64 * Chip8_RUNNING : The emulator is loading and
65 * executing instructions
66 * Chip8_RUN_INSTRUCTION: The emulator will run only one
67 * instruction
68 * Chip8_PAUSED : The emulator is paused and will
69 * only respond to linux
70 */
71
72 typedef enum {
73     Chip8_RUNNING,
74     Chip8_RUN_INSTRUCTION,
75     Chip8_PAUSED
76 } Chip8_STATE;
77
78 /**
79 * STACK_OP defines the behavior of the stack
80 *
81 * STACK_POP : Pop the stack and write out value
82 * STACK_PUSH : Push the input onto the stack
83 * STACK_HOLD : Do nothing
84 */
85
86 typedef enum {
87     STACK_POP,
88     STACK_PUSH,
89     STACK_HOLD
90 } STACK_OP;
91
92 parameter NEXT_PC_WRITE_STAGE = 32'd12;
93 parameter CPU_CYCLE_LENGTH = 32'd50000;
94 parameter FRAMEBUFFER_REFRESH_HOLD = 32'd200000;
95 parameter COPY_THRESHOLD = 32'd500000;
96
97 'endif

```

## 7.2 Testbenches

### 7.2.1 Chip8\_CPU\_6xkk\_7xkk.sv

```
1  `timescale 1ns/100ps
2
3  `include "../enums.svh"
4
5  /**
6   * Test to make sure that after an instruction happens,
7   * all output values get reset to their defaults
8   */
9  task automatic test_resets(ref logic[31:0] stage, ref int
10    → total, ref int failed,
11    ref logic delay_timer_WE, sound_timer_WE,
12    ref logic[7:0] delay_timer_writedata,
13    → sound_timer_writedata,
14    //ref PC_SRC pc_src,
15    ref logic[11:0] PC_writedata,
16    ref logic reg_WE1, reg_WE2,
17    ref logic[3:0] reg_addr1, reg_addr2,
18    ref logic[7:0] reg_writedata1, reg_writedata2,
19    ref logic mem_WE1, mem_WE2,
20    ref logic[11:0] mem_addr1, mem_addr2,
21    ref logic[ 7:0] mem_writedata1, mem_writedata2,
22    ref logic reg_I_WE,
23    ref logic[15:0] reg_I_writedata,
24    ref logic sp_push, sp_pop,
25    ref logic [4:0] fb_addr_y, //max val = 31
26    ref logic [5:0] fb_addr_x, //max val = 63
27    ref logic fb_writedata,fb_WE, fbreset,
28    ref logic halt_for_keypress);
29  #1ns;
30  assert(
31    delay_timer_WE          == 1'b0 &
32    sound_timer_WE          == 1'b0 &
33    delay_timer_writedata   == 8'b0 &
```

```

32      sound_timer_writedata    == 8'b0 &
33      //                      == PC_SRC_NEXT &
34      pc_src                  == 12'b0 &
35      reg_WE1                 == 1'b0 &
36      reg_WE2                 == 1'b0 &
37      reg_addr1                == 4'b0 &
38      reg_addr2                == 4'b0 &
39      reg_writedata1           == 8'b0 &
40      reg_writedata2           == 8'b0 &
41      mem_WE1                 == 1'b0 &
42      mem_WE2                 == 1'b0 &
43      mem_addr1                == 12'h0 &
44      mem_addr2                == 12'h0 &
45      mem_writedata1           == 8'h0 &
46      mem_writedata2           == 8'h0 &
47      reg_I_WE                 == 1'b0 &
48      reg_I_writedata          == 16'h0 &
49      sp_push                  == 1'b0 &
50      sp_pop                  == 1'b0 &
51      fb_addr_y               == 5'h0 &
52      fb_addr_x               == 6'h0 &
53      fb_writedata             == 1'b0 &
54      fb_WE                   == 1'b0 &
55      fbreset                  == 1'b0 &
56      halt_for_keypress        == 1'b0) begin
57      $display("All outputs reset to their defaults");
58      total = total + 1;
59  end else begin
60      $display("Outputs were NOT reset to their defaults.
61      ↳ Current stage: %d",stage);
62      failed = failed + 1;
63  end
64
65
66  /**
67   * Tests to make sure CPU outputs proper
68   * request signafor 7xkk by sending 7E54;

```

```

69  */
70  task automatic test7xkk(ref logic cpu_clk,
71          ref logic[15:0] instruction,
72          ref logic[31:0] stage,
73          ref int total,
74          ref int failed,
75          ref logic reg_WE1, reg_WE2,
76          ref logic[3:0] reg_addr1, reg_addr2,
77          ref logic[7:0] reg_writedata1,
78          → reg_writedata2,
79          reg_readdata1);
80
81  repeat (2) @posedge cpu_clk;
82  stage = 32'h0;
83  instruction = 16'h7EF0;
84  reg_readdata1 = 8'h01;
85
86  wait(stage == 32'h2); #1ns;
87  assert(reg_addr1 == instruction[11:8])
88  else $display("Instruction 7xkk failed with instruction
     → %h in stage %d", instruction, stage);
89
90  wait(stage == 32'h3); #1ns;
91  assert(reg_addr1 == instruction[11:8] &
92          (reg_writedata1 == (reg_readdata1 +
93          → instruction[7:0])) &
94          reg_WE1 == 1'b1
95          //cannot check alu_in1, alu_in2, alu_cmd
96          → without internal access
97          ) begin
98      total = total + 1;
99      $display("Instruction 7xkk passed with instruction
     → %h", instruction);
100 end

```

```

101
102     wait(stage == 32'h4);
103
104
105 endtask
106
107 /**
108 * Tests to make sure instruction 6xkk works
109 * by testing 61F0.
110 */
111 task automatic test6xkk(ref logic cpu_clk,
112                         ref logic[15:0] instruction,
113                         ref logic[31:0] stage,
114                         ref int total,
115                         ref int failed,
116                         ref logic reg_WE1, reg_WE2,
117                         ref logic[3:0] reg_addr1, reg_addr2,
118                         ref logic[7:0] reg_writedata1,
119                         → reg_writedata2);
120     repeat (2) @ (posedge cpu_clk);
121     stage = 32'h0;
122     repeat (1) @ (posedge cpu_clk);
123
124     instruction = 16'h61F0;
125
126     wait(instruction && 16'h61F0 && stage == 32'h2);
127     #1ns;
128     assert((reg_addr1 == 4'h1) && (reg_writedata1 == 8'hF0) &&
129             → (reg_WE1 == 1'h1))begin
130         $display("6xkk passed with instr %h", instruction);
131         total = total + 1;
132     end else begin
133         $display("6xkk FAILED with instr %h", instruction);
134         failed = failed + 1;
135     end
136     wait(stage == 32'h3);
137
138 endtask

```

```

137
138 module Chip8_CPU_6xkk_7xkk( );
139
140     logic cpu_clk;
141     logic[15:0] instruction;
142     logic[7:0]    reg_readdata1, reg_readdata2,
143                 mem_readdata1, mem_readdata2;
144     logic[15:0] reg_I_readdata;
145     logic[7:0] delay_timer_readdata;
146
147     logic key_pressed;
148     logic[3:0] key_press;
149
150     logic[11:0] PC_readdata;
151
152     logic[31:0] stage;
153
154     logic fb_readdata;
155
156     Chip8_STATE top_level_state;
157
158     logic delay_timer_WE, sound_timer_WE;
159     logic[7:0] delay_timer_writedata, sound_timer_writedata;
160
161     PC_SRC pc_src;
162     logic[11:0] PC_writedata;
163
164     logic reg_WE1, reg_WE2;
165     logic[3:0] reg_addr1, reg_addr2;
166     logic[7:0] reg_writedata1, reg_writedata2;
167
168     logic mem_WE1, mem_WE2;
169     logic[11:0] mem_addr1, mem_addr2;
170     logic[7:0] mem_writedata1, mem_writedata2;
171
172     logic reg_I_WE;
173     logic[15:0] reg_I_writedata;
174     logic sp_push, sp_pop;

```

```

175
176     logic [4:0]      fb_addr_y;//max val = 31
177     logic [5:0]      fb_addr_x;//max val = 63
178     logic           fb_writedata, //data to write to address.
179                           fb_WE, //enable writing to
180                           → address
181                           fbreset;
182
183     logic halt_for_keypress;
184     int total = 0;
185     int failed = 0;
186
187     Chip8_CPU dut(.*);
188
189
190
191     initial begin
192         cpu_clk = 0;
193         stage = 32'b0;
194         forever begin
195             #20ns cpu_clk = 1;
196             stage = stage + 1;
197             #20ns cpu_clk = 0;
198         end
199     end
200
201     initial begin
202         $display("Starting test tasks.");
203         test6xkk(cpu_clk, instruction,
204                   → stage,total,failed,reg_WE1, reg_WE2,
205                   reg_addr1, reg_addr2,reg_writedata1,
206                   → reg_writedata2);
207
208         test_resets(stage, total,failed,delay_timer_WE,
209                     → sound_timer_WE,
210                     delay_timer_writedata, sound_timer_writedata,
211                     → /*PC_SRC pc_src,*/

```

```

208     PC_writedata,reg_WE1, reg_WE2,reg_addr1,
209     ↳ reg_addr2,reg_writedata1, reg_writedata2,
210     mem_WE1, mem_WE2, mem_addr1,
211     ↳ mem_addr2,mem_writedata1, mem_writedata2,
212     reg_I_WE,reg_I_writedata,sp_push,
213     ↳ sp_pop,fb_addr_y,fb_addr_x,
214     fb_writedata,fb_WE, fbreset,halt_for_keypress);

215
216     test7xkk(cpu_clk, instruction,
217     ↳ stage,total,failed,reg_WE1, reg_WE2,
218             reg_addr1, reg_addr2, reg_writedata1,
219             ↳ reg_writedata2, reg_readdata1);

220     test_resets(stage, total,failed,delay_timer_WE,
221     ↳ sound_timer_WE,
222         delay_timer_writedata, sound_timer_writedata,
223         ↳ /*PC_SRC pc_src,*/
224         PC_writedata,reg_WE1, reg_WE2,reg_addr1,
225         ↳ reg_addr2,reg_writedata1, reg_writedata2,
226         mem_WE1, mem_WE2, mem_addr1,
227         ↳ mem_addr2,mem_writedata1, mem_writedata2,
228         reg_I_WE,reg_I_writedata,sp_push,
229         ↳ sp_pop,fb_addr_y,fb_addr_x,
230         fb_writedata,fb_WE, fbreset,halt_for_keypress);

231
232     $display("Total number of tests passed: %d", total);
233     $display("Total number of tests failed: %d", failed);
234 end
235
236
237
238 endmodule

```

## 7.2.2 Chip8\_CPU\_big\_testbench.sv

```

1  /**
2   * Author: Levi Oliver
3   * This code tests instructions

```

```

4   * 00e0 -- clear screen
5   * 6xkk -- load kk into Vx
6   * 7xkk -- sets Vx = Vx + kk
7   * Dxyn -- draws sprite! see instruction description in
    ↳ Chip8_CPU.sv
8   * Fx29 -- sets I to memory address of sprite representing
    ↳ value in Vx
9   * Fx33 -- stores BCD value for Vx in memory at I/I+1/I+2 :::
    ↳ h/t/o
10  */
11
12
13
14 `timescale 1ns/100ps
15
16 `include "../enums.svh"
17
18 /**
19  * Test to make sure that after an instruction happens,
20  * all output values get reset to their defaults
21  */
22 task automatic test_resets(ref logic cpu_clk, ref logic[31:0]
    ↳ stage, ref int total, ref int failed,
    ref logic delay_timer_WE, sound_timer_WE,
    ref logic[7:0] delay_timer_writedata,
    ↳ sound_timer_writedata,
    //ref PC_SRC pc_src,
    ref logic[11:0] PC_writedata,
    ref logic reg_WE1, reg_WE2,
    ref logic[3:0] reg_addr1, reg_addr2,
    ref logic[7:0] reg_writedata1, reg_writedata2,
    ref logic mem_WE1, mem_WE2,
    ref logic[11:0] mem_addr1, mem_addr2,
    ref logic[7:0] mem_writedata1, mem_writedata2,
    ref logic reg_I_WE,
    ref logic[15:0] reg_I_writedata,
    ref logic sp_push, sp_pop,
    ref logic [4:0] fb_addr_y, //max val = 31

```

```

37    ref logic [5:0]    fb_addr_x, //max val = 63
38    ref logic fb_writedata,fb_WE, fbreset,
39    ref logic halt_for_keypress);
40    #3ns;
41    wait(cpu_clk == 1'b0);
42    assert(
43        delay_timer_WE          == 1'b0 &
44        sound_timer_WE         == 1'b0 &
45        delay_timer_writedata == 8'b0 &
46        sound_timer_writedata == 8'b0 &
47        pc_src                == PC_SRC_NEXT &
48        PC_writedata           == 12'b0 &
49        reg_WE1                == 1'b0 &
50        reg_WE2                == 1'b0 &
51        reg_addr1              == 4'b0 &
52        reg_addr2              == 4'b0 &
53        reg_writedata1         == 8'b0 &
54        reg_writedata2         == 8'b0 &
55        mem_WE1                == 1'b0 &
56        mem_WE2                == 1'b0 &
57        mem_addr1              == 12'h0 &
58        mem_addr2              == 12'h0 &
59        mem_writedata1         == 8'h0 &
60        mem_writedata2         == 8'h0 &
61        reg_I_WE               == 1'b0 &
62        reg_I_writedata        == 16'h0 &
63        fb_addr_y              == 5'h0 &
64        fb_addr_x              == 6'h0 &
65        fb_writedata            == 1'b0 &
66        fb_WE                  == 1'b0 &
67        fbreset                == 1'b0 &
68        halt_for_keypress      == 1'b0) begin
69        $display("All outputs reset to their defaults");
70        total = total + 1;
71    end else begin
72        $display("Outputs were NOT reset to their defaults.
73        ↳ Current stage: %d",stage);
74        failed = failed + 1;

```

```

74    end
75    repeat (2) @ (posedge cpu_clk);
76
77  endtask
78
79  task automatic testFx33 (ref logic cpu_clk,
80                          ref logic [15:0] instruction,
81                          ref logic [31:0] stage,
82                          ref int          total, failed,
83                          ref logic [3:0] reg_addr1,
84                          ref logic [7:0] reg_readdata1,
85                          ref logic [11:0] mem_addr1,
86                          ref logic [7:0] mem_writedata1,
87                          ref logic          mem_WE1,
88                          ref logic [15:0] reg_I_readdata);
89
90  instruction = 16'hFe33;
91  stage = 32'b0;
92
93  wait (stage == 32'h2); #1ns;
94  assert (reg_addr1 == instruction[11:8]);
95
96  reg_readdata1 = 8'd195; //1100 0011
97  reg_I_readdata = 16'h03F2;
98
99  wait (stage == 32'h3); #1ns;
100  assert (reg_addr1 == instruction[11:8] &
101      → mem_addr1 == reg_I_readdata[11:0] &
102          mem_addr1 == reg_I_readdata[11:0] &
103              → mem_writedata1 == 8'd1 & mem_WE1)
104 else begin
105     $display ("Improper BCD conversion. \n\tGiven value:
106      → %d\n\tCalculated hundreds place:
107      → %d", reg_readdata1, mem_writedata1);
108 end
109
110
111  wait (stage == 32'h4); #1ns;

```

```

108      assert(reg_addr1==instruction[11:8]  &
109          → mem_addr1==(1+reg_I_readdata[11:0]) &
110          → mem_writedata1==8'd9 & mem_WE1)
111  else begin
112      $display("Improper BCD conversion. \n\tGiven value:
113          → %d\n\tCalculated tens place: %d",reg_readdata1,
114          → mem_writedata1);
115  end
116
117  wait(stage == 32'h5);#1ns;
118  assert(mem_addr1==(2+reg_I_readdata[11:0]) &
119          → mem_writedata1==8'd5 & mem_WE1) begin
120      total = total + 1;
121      $display("Instruction Fx33 (store Vx as BCD in I/+1/+2
122          → in mem) is a success.");
123  end else begin
124      $display("Improper BCD conversion. \n\tGiven value:
125          → %d\n\tCalculated ones place: %d",reg_readdata1,
126          → mem_writedata1);
127  end
128
129  wait(stage == 32'h6); #1ns;
130 endtask
131
132
133
134 task automatic testFx29(ref logic cpu_clk,
135                         ref logic[15:0] instruction,
136                         ref logic[31:0] stage,
137                         ref int          total, failed,
138                         ref logic[ 3:0] reg_addr1,
139                         ref logic[ 7:0] reg_readdata1,
140                         ref logic[15:0] reg_I_writedata,
141                         ref logic          reg_I_WE);
142
143
144     stage = 32'b0;
145     instruction = 16'hFC29;
146
147     wait(stage == 32'h2);#1ns;

```

```

138 assert(reg_addr1==instruction[11:8])
139 else begin
140     failed = failed + 1;
141     $display("FX29 FAILED IN STAGE 2.");
142 end
143
144 reg_readdata1 = 8'h0E;
145
146 wait(stage == 32'h3);#1ns;
147 assert(reg_I_writedata == 16'd70 & reg_I_WE == 1'b1)
148    begin
149        $display("Fx29 (set I to Font by Vx) works!");
150        total = total + 1;
151    end else begin
152        $display("Fx29 failed in stage 3");
153    end
154
155
156
157 endtask
158
159
160 task automatic testDdyn(ref logic cpu_clk,
161                         ref logic[15:0] instruction,
162                         ref logic[31:0] stage,
163                         ref int total,
164                         ref int failed,
165                         ref logic[3:0] reg_addr1, reg_addr2,
166                         ref logic reg_WE1, reg_WE2,
167                         ref logic[7:0] reg_readdata1,
168                         → reg_readdata2,
169                         ref logic[7:0] mem_readdata1,
170                         → mem_writedata1,
171                         ref logic mem_WE1,
172                         ref logic[11:0] mem_addr1,
173                         ref logic[5:0] fb_addr_x,
174                         ref logic[4:0] fb_addr_y,

```

```

173          ref logic fb_writedata, fb_readdata,
174          ↵ fb_WE,
175          ref logic[15:0] reg_I_readdata,
176          ref logic bit_overwritten, isDrawing);
177
178 instruction = 16'hd392;
179 stage = 32'h0;
180 reg_I_readdata = 16'h0F0F;
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203

```

```

ref logic fb_writedata, fb_readdata,
    ↵ fb_WE,
ref logic[15:0] reg_I_readdata,
    ref logic bit_overwritten, isDrawing);
instruction = 16'hd392;
stage = 32'h0;
reg_I_readdata = 16'h0F0F;

if((stage == 32'h0) || (stage == 32'h1)) begin
    assert(reg_WE1==1'b0 & reg_WE2==1'b0 &
    ↵ mem_WE1==1'b0 & fb_WE==1'b0)
    else begin
        $display("INSTR DXYN HAS INVALID WRITE_EENABLE
        ↵ VALS BEFORE STAGE 2");
        failed = failed + 1;
    end
end

wait(stage == 32'h2); #1ns;
assert(reg_addr1==instruction[11:8] &
    ↵ reg_addr2==instruction[7:4] &
    mem_addr1==reg_I_readdata & reg_WE1==1'b0 &
    ↵ reg_WE2==1'b0 &
    mem_WE1==1'b0 & fb_WE==1'b0 & isDrawing)
else begin
    $display("INSTR DXYN HAS FAILED TO SET INITIAL OUTPUT
    ↵ VALS IN STAGE 3");
    failed = failed + 1;
end

fb_readdata = 1'b0;
reg_readdata1 = 8'd3; //write to x = 3
reg_readdata2 = 8'd9; //write to y = 9
                                //that is position x + 64y =
                                ↵ 576
mem_readdata1 = 8'hAF;

```

```

204    wait(stage == 32'h3); #1ns;
205    assert(reg_addr1==instruction[11:8] &
206           reg_addr2==instruction[7:4] &
207               mem_addr1==reg_I_readdata & reg_WE1==1'b0 &
208                   reg_WE2==1'b0 &
209                   mem_WE1==1'b0 & fb_WE==1'b0 & isDrawing)
210    else begin
211        $display("INSTR DXYN HAS FAILED TO HOLD VALS FROM
212                  STATE 2 IN STATE 3\n\tEx--mem_addr1=
213                  %h",mem_addr1);
214        failed = failed + 1;
215    end
216
217    wait(stage == 32'd15); #1ns;
218    assert(reg_addr1==instruction[11:8] &
219           reg_addr2==instruction[7:4] &
220               mem_addr1==reg_I_readdata & reg_WE1==1'b0 &
221                   reg_WE2==1'b0 &
222                   mem_WE1==1'b0 & fb_WE==1'b0 & isDrawing)
223    else begin
224        $display("INSTR DXYN HAS FAILED TO HOLD VALS FROM
225                  STATE 2 IN STATE 15\n\tEx--mem_addr1=
226                  %h",mem_addr1);
227        failed = failed + 1;
228    end
229
230    wait(stage == 32'd17); #1ns;
231    assert(reg_addr1==instruction[11:8] &
232           reg_addr2==instruction[7:4] &
233               mem_addr1==reg_I_readdata & fb_WE==1'b1 &
234                   fb_addr_x==reg_readdata1 &
235                       fb_addr_y==reg_readdata2 &
236                       fb_writedata==1'b1 & bit_overwritten==1'b0 &
237                           isDrawing)
238    else begin

```

```

230      $display("INSTR DXYN FAILED IN ITS FIRST WRITE
231          ↳ STAGE");
232      failed = failed + 1;
233  end
234  wait(stage == 32'd18); #1ns;
235  assert(reg_addr1==instruction[11:8] &
236      ↳ reg_addr2==instruction[7:4] &
237          mem_addr1==reg_I_readdata & fb_WE==1'b0 &
238          fb_addr_x==(reg_readdata1+1) &
239              ↳ fb_addr_y==reg_readdata2 &
240                  bit_overwritten==1'b0 & isDrawing)
241 else begin
242     $display("INSTR DXYN FAILED IN STAGE 18");
243     failed = failed + 1;
244 end
245 fb_readdata = 1'b1;
246 wait(stage == 32'd31); #1ns;
247 assert(reg_addr1==instruction[11:8] &
248      ↳ reg_addr2==instruction[7:4] &
249          mem_addr1==reg_I_readdata & fb_WE==1'b1 &
250          fb_addr_x==(reg_readdata1+7) &
251              ↳ fb_addr_y==reg_readdata2 &
252                  bit_overwritten==1'b1 & isDrawing)
253 else begin
254     $display("INSTR DXYN FAILED IN STAGE 31");
255     failed = failed + 1;
256 end
257 fb_readdata = 1'b0;

258
259 wait(stage == 32'd32); #3ns;
260 assert(reg_addr1==instruction[11:8] &
261      ↳ reg_addr2==instruction[7:4] &
262          mem_addr1==(reg_I_readdata+1) & fb_WE==1'b0 &
263          fb_addr_x==(reg_readdata1) &
264              ↳ fb_addr_y==(reg_readdata2+1) &
265                  bit_overwritten==1'b0 & isDrawing)
266 else begin

```

```

261      $display("INSTR DXYN FAILED IN STAGE
262      ~ 32\n\treg_readdata2=%h\n\tfb_addr_y=%h",reg_readdata2,fb_addr_y);
263      $display("\tmem_addr1=%h\n\treg_I_readdata=%h",mem_addr1,reg_I_readdata);
264      failed = failed + 1;
265
266      mem_readdata1 = 8'b1101000;
267
268      wait(stage == 32'd33); #1ns;
269      assert(reg_addr1==instruction[11:8] &
270             ~ reg_addr2==instruction[7:4] &
271             mem_addr1==(reg_I_readdata+1) & fb_WE==1'b1 &
272             fb_addr_x==(reg_readdata1) &
273             ~ fb_addr_y==(reg_readdata2+1) &
274             bit_overwritten==1'b0 & isDrawing)
275      else begin
276          $display("INSTR DXYN FAILED IN STAGE
277          ~ 33\n\treg_readdata2=%h\n\tfb_addr_y=%h",reg_readdata2,fb_addr_y);
278          $display("\tmem_addr1=%h\n\treg_I_readdata=%h",mem_addr1,reg_I_readdata);
279          failed = failed + 1;
280      end
281
282      wait(stage == 32'd39); #1ns;
283      assert(reg_addr1==instruction[11:8] &
284             ~ reg_addr2==instruction[7:4] &
285             mem_addr1==(reg_I_readdata+1) & fb_WE==1'b1 &
286             fb_addr_x==(reg_readdata1+8'h3) &
287             ~ fb_addr_y==(reg_readdata2+1) &
288             fb_writedata==1'b1 & bit_overwritten==1'b0 &
289             ~ isDrawing)
290      else begin
291          $display("INSTR DXYN FAILED IN STAGE
292          ~ 39\n\treg_readdata2=%h\n\tfb_addr_y=%h",reg_readdata2,fb_addr_y);
293          $display("\tmem_addr1=%h\n\treg_I_readdata=%h",mem_addr1,reg_I_readdata);
294          failed = failed + 1;
295      end
296
297      wait(stage == 32'd47); #1ns;

```

```

291      assert(reg_addr1==instruction[11:8] &
292          → reg_addr2==instruction[7:4] &
293              mem_addr1==(reg_I_readdata+1) & fb_WE==1'b1 &
294                  fb_addr_x==(reg_readdata1+8'h7) &
295                      → fb_addr_y==(reg_readdata2+1) &
296                          fb_writedata==1'b0 & bit_overwritten==1'b0 &
297                              → isDrawing)
298
299 else begin
300     $display("INSTR DXYN FAILED IN STAGE
301             → 47\n\treg_readdata2=%h\n\tfb_addr_y=%h",reg_readdata2,fb_addr_y);
302     $display("\tmem_addr1=%h\n\treg_I_readdata=%h",mem_addr1,reg_I_readdata);
303     failed = failed + 1;
304 end
305
306 wait(stage == 32'd48); #1ns;
307 assert(fb_WE==1'b0 & bit_overwritten==1'b0 & isDrawing)
308
309 else begin
310     failed = failed + 1;
311     $display("INSTR DXYN FAILED IN STAGE: %d", stage);
312 end
313
314
315 wait(stage == 32'd61); #1ns;
316 assert(fb_WE==1'b0 & bit_overwritten==1'b0 & isDrawing)
317
318 → begin
319     $display("Dxyn draw sprite works!! :D");
320     total = total + 1;
321 end else begin
322     $display("INSTR DXYN FAILED IN STAGE: %d", stage);
323     failed = failed + 1;
324 end
325
326
327 $display("W000000BL");
328 wait(stage == 32'd100);#1ns;
329 stage = 32'b0;
330
331 endtask
332
333 /**

```

```

324  * Tests to make sure CPU outputs proper
325  * request signafor 7xkk by sending 7E54;
326  */
327 task automatic test7xkk(ref logic cpu_clk,
328                     ref logic[15:0] instruction,
329                     ref logic[31:0] stage,
330                     ref int total,
331                     ref int failed,
332                     ref logic reg_WE1, reg_WE2,
333                     ref logic[3:0] reg_addr1, reg_addr2,
334                     ref logic[7:0] reg_writedata1,
335                     → reg_writedata2,
336                               reg_readdata1);
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355

```

---

```

* request signafor 7xkk by sending 7E54;
*/
task automatic test7xkk(ref logic cpu_clk,
                     ref logic[15:0] instruction,
                     ref logic[31:0] stage,
                     ref int total,
                     ref int failed,
                     ref logic reg_WE1, reg_WE2,
                     ref logic[3:0] reg_addr1, reg_addr2,
                     ref logic[7:0] reg_writedata1,
                     → reg_writedata2,
                               reg_readdata1);

repeat (2) @posedge cpu_clk;
stage = 32'h0;
instruction = 16'h7EF0;
reg_readdata1 = 8'h01;

wait(stage == 32'h2); #1ns;
assert(reg_addr1 == instruction[11:8])
else $display("Instruction 7xkk failed with instruction
→ %h in stage %d", instruction, stage);

wait(stage == 32'h3); #1ns;
assert(reg_addr1 == instruction[11:8] &
       (reg_writedata1 == (reg_readdata1 +
                           → instruction[7:0])) &
       reg_WE1 == 1'b1
//cannot check alu_in1, alu_in2, alu_cmd
       → without internal access
) begin
      total = total + 1;
      $display("Instruction 7xkk passed with instruction
→ %h", instruction);
end else begin
      failed = failed + 1;

```

```

356     $display("Instruction 7xkk failed with instruction %h
357         ↳ in stage %d", instruction, stage);
358 end
359
360
361
362 endtask
363
364 /**
365 * Tests to make sure instruction 6xkk works
366 * by testing 61F0.
367 */
368 task automatic test6xkk(ref logic cpu_clk,
369                         ref logic[15:0] instruction,
370                         ref logic[31:0] stage,
371                         ref int total,
372                         ref int failed,
373                         ref logic reg_WE1, reg_WE2,
374                         ref logic[3:0] reg_addr1, reg_addr2,
375                         ref logic[7:0] reg_writedata1,
376                         ↳ reg_writedata2);
377 repeat (2) @ (posedge cpu_clk);
378 stage = 32'h0;
379
380 instruction = 16'h61F0;
381
382 wait(instruction && 16'h61F0 && stage == 32'h2);
383 #1ns;
384 assert((reg_addr1 == 4'h1) && (reg_writedata1 == 8'hF0) &&
385         ↳ (reg_WE1 == 1'h1))begin
386     $display("6xkk passed with instr %h", instruction);
387     total = total + 1;
388 end else begin
389     $display("6xkk FAILED with instr %h", instruction);
390     failed = failed + 1;
391 end

```

```

391     wait(stage == 32'h3);
392
393 endtask
394
395
396 task automatic test00E0(ref logic cpu_clk,
397                         ref logic[15:0] instruction,
398                         ref logic[31:0] stage,
399                         ref int total, failed,
400                         ref logic fb_WE, fb_writedata, fbreset,
401                         ref logic[5:0] fb_addr_x,
402                         ref logic[4:0] fb_addr_y);
403
404     stage = 32'h0;
405     instruction = 16'h00e0;
406
407     wait(stage == 32'h2); #1ns;
408     assert(fbreset == 1'b1)
409     else begin
410         failed = failed + 1;
411         $display("INSTR 00E0: fbreset NEVER SET HIGH.");
412     end
413
414     wait(stage == 32'h3); #1ns;
415     assert(fb_WE==1'b1 & fb_writedata==1'b0 & fb_addr_x==0
416           & fb_addr_y==0 & fbreset==1'b0)
417     else begin
418         failed = failed + 1;
419         $display("INSTR 00E0: Did not start clearing at
420             (x,y)=(0, 0). stage: %h", stage);
421     end
422     if(stage > 32'h2 & stage < 32'd28189 & fbreset==1'b0)
423         begin
424             assert(fb_WE==1'b1 & fb_writedata==1'b0)
425             else begin
426                 failed = failed + 1;
427                 $display("INSTR 00E0: failed in stage: %h",
428                     stage);

```

```

425     end
426
427     wait(stage==32'd8188);#1ns;
428     assert(fb_WE==1'b1 & fb_writedata==1'b0 &
429             (&fb_addr_x) & (&fb_addr_y) & fbreset==1'b0)
430             begin
431                 $display("00e0 clear screen success!");
432                 total = total + 1;
433             end else begin
434                 failed = failed + 1;
435                 $display("INSTR 00E0: x and y addresses of clear
436                         never reach
437                         63x31\n\ttx=%h\n\ty=%h\stage=%d=%b",fb_addr_x,fb_addr_y,stage,stag
438             end
439
440         end
441
442         wait(stage == 32'd8189); #1ns;
443
444     endtask
445
446 module Chip8_CPU_big_testbench( );
447
448     logic cpu_clk;
449     logic[15:0] instruction;
450     logic[7:0]    reg_readdata1, reg_readdata2,
451                 mem_readdata1, mem_readdata2;
452     logic[15:0] reg_I_readdata;
453     logic[7:0] delay_timer_readdata;
454
455     logic key_pressed;
456     logic[3:0] key_press;
457
458     logic[11:0] PC_readdata;
459
460     logic[31:0] stage;
461
462     logic fb_readdata;

```

```

459
460     Chip8_STATE top_level_state;
461
462     logic delay_timer_WE, sound_timer_WE;
463     logic[7:0] delay_timer_writedata, sound_timer_writedata;
464
465     PC_SRC pc_src;
466     logic[11:0] PC_writedata;
467
468     logic reg_WE1, reg_WE2;
469     logic[3:0] reg_addr1, reg_addr2;
470     logic[7:0] reg_writedata1, reg_writedata2;
471
472     logic mem_WE1, mem_WE2;
473     logic[11:0] mem_addr1, mem_addr2;
474     logic[7:0] mem_writedata1, mem_writedata2;
475
476     logic reg_I_WE;
477     logic[15:0] reg_I_writedata;
478     logic sp_push, sp_pop;
479
480     logic [4:0]    fb_addr_y;//max val = 31
481     logic [5:0]    fb_addr_x;//max val = 63
482     logic          fb_writedata, //data to write to addresse.
483                           fb_WE, //enable writing to
484                           → address
485                           fbreset, isDrawing;
486
487     logic halt_for_keypress;
488
489     logic stk_reset;
490     STACK_OP stk_op;
491     logic[15:0] stk_writedata;
492     logic bit_overwritten;
493     logic mem_request;
494
495     int total = 0;
496     int failed = 0;

```

```

496
497
498 Chip8_CPU dut(.*) ;
499
500
501
502 initial begin
503     cpu_clk = 0;
504     stage = 32'b0;
505     forever begin
506         #20ns cpu_clk = 1;
507         stage = stage + 1;
508         #20ns cpu_clk = 0;
509     end
510 end
511
512 initial begin
513     $display("Starting test tasks.");
514     test6xkk(cpu_clk, instruction,
515             → stage,total,failed,reg_WE1, reg_WE2,
516                 reg_addr1, reg_addr2,reg_writedata1,
517                 → reg_writedata2);
518
519     test_resets(cpu_clk, stage,
520             → total,failed,delay_timer_WE, sound_timer_WE,
521                 delay_timer_writedata, sound_timer_writedata,
522                 → /*PC_SRC pc_src,*/
523                 PC_writedata,reg_WE1, reg_WE2,reg_addr1,
524                 → reg_addr2,reg_writedata1, reg_writedata2,
525                 mem_WE1, mem_WE2, mem_addr1,
526                 → mem_addr2,mem_writedata1, mem_writedata2,
527                 reg_I_WE,reg_I_writedata,sp_push,
528                 → sp_pop,fb_addr_y,fb_addr_x,
529                 fb_writedata,fb_WE, fbreset,halt_for_keypress);
530
531
532     test7xkk(cpu_clk, instruction,
533             → stage,total,failed,reg_WE1, reg_WE2,

```

```

525           reg_addr1, reg_addr2, reg_writedata1,
526           ↵   reg_writedata2, reg_readdata1);

527     test_resets(cpu_clk, stage,
528     ↵   total,failed,delay_timer_WE, sound_timer_WE,
529     delay_timer_writedata, sound_timer_writedata,
530     ↵   /*PC_SRC pc_src,*/
531     PC_writedata,reg_WE1, reg_WE2,reg_addr1,
532     ↵   reg_addr2,reg_writedata1, reg_writedata2,
533     mem_WE1, mem_WE2, mem_addr1,
534     ↵   mem_addr2,mem_writedata1, mem_writedata2,
535     reg_I_WE,reg_I_writedata,sp_push,
536     ↵   sp_pop,fb_addr_y,fb_addr_x,
537     fb_writedata,fb_WE, fbreset,halt_for_keypress);

538     testDxyn(cpu_clk, instruction, stage,total,failed,
539     reg_addr1, reg_addr2, reg_WE1, reg_WE2,
540     reg_readdata1,
541     ↵   reg_readdata2,mem_readdata1,
542     mem_writedata1,mem_WE1, mem_addr1,
543     ↵   fb_addr_x,fb_addr_y,
544     fb_writedata, fb_readdata,
545     ↵   fb_WE,reg_I_readdata,
546     ↵   bit_overwritten,isDrawing);

547     test_resets(cpu_clk, stage,
548     ↵   total,failed,delay_timer_WE, sound_timer_WE,
549     delay_timer_writedata, sound_timer_writedata,
550     ↵   /*PC_SRC pc_src,*/
551     PC_writedata,reg_WE1, reg_WE2,reg_addr1,
552     ↵   reg_addr2,reg_writedata1, reg_writedata2,
553     mem_WE1, mem_WE2, mem_addr1,
554     ↵   mem_addr2,mem_writedata1, mem_writedata2,
555     reg_I_WE,reg_I_writedata,sp_push,
556     ↵   sp_pop,fb_addr_y,fb_addr_x,
557     fb_writedata,fb_WE, fbreset,halt_for_keypress);

```

```

547     test00E0(cpu_clk,
548         ↳ instruction,stage,total,failed,fb_WE,
549         ↳ fb_writedata,fbreset,fb_addr_x,fb_addr_y);
550
551     test_resets(cpu_clk, stage,
552         ↳ total,failed,delay_timer_WE, sound_timer_WE,
553         ↳ delay_timer_writedata, sound_timer_writedata,
554         ↳ /*PC_SRC pc_src,*/
555         ↳ PC_writedata,reg_WE1, reg_WE2,reg_addr1,
556         ↳ reg_addr2,reg_writedata1, reg_writedata2,
557         ↳ mem_WE1, mem_WE2, mem_addr1,
558         ↳ mem_addr2,mem_writedata1, mem_writedata2,
559         ↳ reg_I_WE,reg_I_writedata,sp_push,
560         ↳ sp_pop,fb_addr_y,fb_addr_x,
561         ↳ fb_writedata,fb_WE, fbreset,halt_for_keypress);
562
563     testFx29(
564         ↳ cpu_clk,instruction,stage,total,failed,reg_addr1,reg_readdata1,reg_I-
565
566     test_resets(cpu_clk, stage,
567         ↳ total,failed,delay_timer_WE, sound_timer_WE,
568         ↳ delay_timer_writedata, sound_timer_writedata,
569         ↳ /*PC_SRC pc_src,*/
570         ↳ PC_writedata,reg_WE1, reg_WE2,reg_addr1,
571         ↳ reg_addr2,reg_writedata1, reg_writedata2,
572         ↳ mem_WE1, mem_WE2, mem_addr1,
573         ↳ mem_addr2,mem_writedata1, mem_writedata2,
574         ↳ reg_I_WE,reg_I_writedata,sp_push,
575         ↳ sp_pop,fb_addr_y,fb_addr_x,
576         ↳ fb_writedata,fb_WE, fbreset,halt_for_keypress);
577
578     testFx33(cpu_clk,instruction,stage,total,
579         ↳ failed,reg_addr1,reg_readdata1,mem_addr1,mem_writedata1,mem_WE1,
580         ↳ reg_I_readdata);
581
582     test_resets(cpu_clk, stage,
583         ↳ total,failed,delay_timer_WE, sound_timer_WE,

```

```

568     delay_timer_writedata, sound_timer_writedata,
569     → /*PC_SRC pc_src,*/
570     PC_writedata, reg_WE1, reg_WE2, reg_addr1,
571     → reg_addr2, reg_writedata1, reg_writedata2,
572     mem_WE1, mem_WE2, mem_addr1,
573     → mem_addr2, mem_writedata1, mem_writedata2,
574     reg_I_WE, reg_I_writedata, sp_push,
575     → sp_pop, fb_addr_y, fb_addr_x,
576     fb_writedata, fb_WE, fbreset, halt_for_keypress);

577
578
579 endmodule

```

### 7.2.3 Chip8\_CPU\_testbench.sv

```

1 module Chip8_CPU_testbench ( );
2   logic clk;
3   logic[15:0] instruction;
4   logic[3:0] testIn1, testIn2;
5   wire[7:0] testOut1, testOut2;
6
7 //Initialize module here
8 Chip8_CPU cpu (.cpu_clk(clk), .*);
9
10 initial begin
11   clk = 0;
12   forever
13     #20ns clk = ~clk;
14 end
15
16 initial begin
17   instruction = 16'h6122;
18   repeat (2)

```

```

19          @(posedge clk);
20      instruction = 16'h6020;
21      repeat (2)
22          @(posedge clk);
23      instruction = 16'h8014;
24      repeat (2)
25          @(posedge clk);
26      instruction = 16'h8014;
27      repeat (2)
28          @(posedge clk);
29      instruction = 16'h8013;
30      repeat (2)
31          @(posedge clk);
32      instruction = 16'h8015;
33  end
34 endmodule

```

#### 7.2.4 Chip8\_Top\_test.sv

```

1 module Chip8_Top_test ( );
2     //Garbage variables here
3     logic           clk;
4     logic           reset;
5     logic [31:0]    writedata;
6     logic           write;
7     logic           chipselect;
8     logic [17:0]    address;
9
10    logic [31:0]   data_out;
11    logic [7:0]    VGA_R, VGA_G, VGA_B;
12    logic          VGA_CLK, VGA_HS, VGA_VS, VGA_BLANK_n;
13    logic          VGA_SYNC_n;
14
15    //Initialize module here
16    Chip8_Top(.*);
17
18    initial begin

```

```

19         clk = 0;
20         forever
21             #20ns clk = ~clk;
22         end
23
24     initial begin // Initially set the PC to 200
25         //Reset
26         reset = 0;
27         repeat (2)
28             @(posedge clk);
29         reset = 1;
30         repeat (2)
31             @(posedge clk);
32         reset = 0;
33     end
34 endmodule

```

### 7.2.5 delay\_timer\_testbench.sv

```

1  ****
2  * Timer Test Bench
3  * GT
4  ****
5
6  module delay_timer_testbench();
7      logic clk;
8      logic clk_60;
9      logic reset;
10     logic [7:0] data;
11     logic write_enable;
12     wire out;
13
14     delay_timer dut(.write_enable, .clk, .clk_60, .data,
15                     .out);
16
17     initial begin
18         clk = 0;

```

```

18     forever
19         #20ns clk = ~clk;
20     end
21
22     initial begin
23         clk_60 = 1'b0;
24         reset = 0;
25         data = 8'b0000_0000;
26         write_enable = 0;
27
28         repeat (8) begin
29             @(posedge clk);
30             clk_60 = ~clk_60;
31         end
32
33         clk_60 = 1'b1;
34         data = 8'b0000_1000;
35         write_enable = 1;
36
37         repeat (2)
38             @(posedge clk);
39         write_enable = 0;
40
41         repeat (64) begin
42             @(posedge clk);
43             clk_60 = ~clk_60;
44         end
45
46     end
47
48 endmodule
49

```

### 7.2.6 Triple\_port\_reg\_file\_test.sv

```

1 | 'timescale 10ns/10ns
2 |

```

```

3  module Triple_port_reg_file_test ( ) ;
4      logic          cpu_clk; //system clock that controls
5          ↳ writing data
6      logic[7:0]      writedata1, writedata2, VFwritedata; //data
7          ↳ to be written to corresponding addresses
8      logic          WE1, WE2, WEVF; //enable writing on
9          ↳ addressed registers
10     logic[3:0]      addr1, addr2; //addresses to write to and
11         ↳ read from
12     logic[7:0]      readdata1, readdata2, VFreaddata; //data
13         ↳ output from addressed registers
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
//Initialize module here
Chip8_register_file tprf (.cpu_clk(cpu_clk), .*);

initial begin
    cpu_clk = 0;
    forever
        #20ns cpu_clk = ~cpu_clk;
end

initial begin
    WE1 = 1;
    writedata1 = 8'hEE;
    addr1 = 4'h0;
    repeat (2)
        @(posedge cpu_clk);
    WE2 = 1;
    writedata2 = 8'h44;
    addr2 = 4'h1;
    repeat (2)
        @(posedge cpu_clk);
    WEVF = 1;
    VFwritedata = 8'hFF;
    repeat (2)
        @(posedge cpu_clk);
    WE1 = 0;

```

```

36      WE2 = 0;
37      WEVF = 0;
38      addr1 = 4'h0;
39      addr2 = 4'h0;
40      repeat (2)
41          @(posedge cpu_clk);
42          WE1 = 1;
43          WE2 = 1;
44          WEVF = 1;
45          addr1 = 4'h5;
46          writedata1 = 8'h5;
47          addr2 = 4'h6;
48          writedata2 = 8'h6;
49          VFwritedata = 8'hFE;
50          repeat (2)
51              @(posedge cpu_clk);
52              WE1 = 0;
53              WE2 = 0;
54              WEVF = 0;
55              addr1 = 4'h0;
56              addr2 = 4'h0;
57              repeat (2)
58                  @(posedge cpu_clk);
59                  addr1 = 4'h5;
60                  addr2 = 4'h6;
61                  repeat (2)
62                      @(posedge cpu_clk);
63                      addr1 = 4'h0;
64                      addr2 = 4'h1;
65      end
66  endmodule

```

### 7.2.7 fb\_testbench.sv

```

1  ****
2  * Stack Test Bench
3  *

```

```

4   * Author: Gabrielle Taylor, Ashley Kling
5
6   //*****
7   //task automatic testReset(logic clk,
8   //                           logic[4:0] fb_addr_y,
9   //                           logic[5:0] fb_addr_x,
10  //                           logic fb_writedata,
11  //                           logic fb_WE,
12  //                           logic reset);
13  //   fb_addr_y = 6'h0000;
14  //   fb_addr_x = 5'h0000;
15  //   fb_writedata = 1'h0;
16  //   fb_WE = 1'h0;
17  //
18  //   reset = 1'h1;
19  //   repeat(2) @(posedge clk);
20  //   reset = 1'h0;
21  //   repeat(2) @(posedge clk);
22  //endtask
23
24 task automatic testWriteOne(ref logic clk,
25                           ref logic[4:0] fb_addr_y,
26                           ref logic[5:0] fb_addr_x,
27                           ref logic fb_writedata,
28                           ref logic fb_WE,
29                           ref logic fb_readdata,
30                           ref logic reset,
31                           ref int total);
32   fb_addr_y = 5'b00001;
33   fb_addr_x = 6'b000001;
34   fb_writedata = 1;
35   fb_WE = 1;
36   repeat(2) @(posedge clk);
37
38   fb_addr_y = 5'b00000;
39   fb_addr_x = 6'b000000;
40   fb_writedata = 0;

```

```

41     fb_WE = 0;
42     repeat(2) @(posedge clk);
43
44     fb_addr_y = 5'b00001;
45     fb_addr_x = 6'b000001;
46     fb_writedata = 0;
47     fb_WE = 0;
48     repeat(2) @(posedge clk);
49
50     repeat(2)
51       @(posedge clk);
52     assert (fb_readdata == 1'h1) begin
53       $display ("WriteOne TEST 1 : PASSED");
54       total = total + 1;
55     end
56     else $error("OR TEST 1 : FAILED (Got %h, Expected 1)",
57               ↪   fb_readdata);
58
59     fb_addr_y = 6'h0000;
60     fb_addr_x = 5'h0000;
61     fb_writedata = 1'h0;
62     fb_WE = 1'h0;
63
64     reset = 1'h1;
65     repeat(2) @(posedge clk);
66     reset = 1'h0;
67     repeat(2) @(posedge clk);
68     //testReset(clk, fb_addr_y, fb_addr_x, fb_writedata,
69     ↪   fb_WE, reset);
70
71   endtask
72
73   task automatic testWriteOneReadElse(ref logic clk,
74                                     ref logic[4:0] fb_addr_y,
75                                     ref logic[5:0] fb_addr_x,
76                                     ref logic fb_writedata,
77                                     ref logic fb_WE,
78
79

```

```

76                               ref logic fb_readdata,
77                               ref logic reset,
78                               ref int total);
79   fb_addr_y = 5'b00001;
80   fb_addr_x = 6'b000001;
81   fb_writedata = 1;
82   fb_WE = 1;
83   repeat(2) @(posedge clk);

84
85   fb_addr_y = 5'b00000;
86   fb_addr_x = 6'b000000;
87   fb_writedata = 0;
88   fb_WE = 0;
89   repeat(2) @(posedge clk);

90
91   repeat(2)
92     @(posedge clk);
93   assert (fb_readdata == 1'h0) begin
94     $display ("WriteOneRead Else TEST 2 : PASSED");
95     total = total + 1;
96   end
97   else $error("OR TEST 2 : FAILED (Got %h, Expected 0)",
98             ~ fb_readdata);

99   fb_addr_y = 6'h0000;
100  fb_addr_x = 5'h0000;
101  fb_writedata = 1'h0;
102  fb_WE = 1'h0;

103
104  reset = 1'h1;
105  repeat(2) @(posedge clk);
106  reset = 1'h0;
107  repeat(2) @(posedge clk);
108  //testReset(clk, fb_addr_y, fb_addr_x, fb_writedata,
109             ~ fb_WE, reset);
110
111 endtask

```

```

111
112
113 task automatic testWriteManyReadMany(ref logic clk,
114                                         ref logic[4:0] fb_addr_y,
115                                         ref logic[5:0] fb_addr_x,
116                                         ref logic fb_writedata,
117                                         ref logic fb_WE,
118                                         ref logic fb_readdata,
119                                         ref logic reset,
120                                         ref int total);
121     repeat(4) @(posedge clk);
122
123         fb_addr_y = 5'b00010;
124         fb_addr_x = 6'b000010;
125         fb_writedata = 1;
126         fb_WE = 1;
127         repeat(2) @(posedge clk);
128
129         fb_addr_y = 5'b00000;
130         fb_addr_x = 6'b000000;
131         fb_writedata = 0;
132         fb_WE = 0;
133         repeat(2) @(posedge clk);
134
135         fb_addr_y = 5'b00100;
136         fb_addr_x = 6'b000100;
137         fb_writedata = 1;
138         fb_WE = 1;
139         repeat(2) @(posedge clk);
140
141         fb_addr_y = 5'b01000;
142         fb_addr_x = 6'b001000;
143         fb_writedata = 1;
144         repeat(2) @(posedge clk);
145
146         fb_addr_y = 5'b10000;
147         fb_addr_x = 6'b010000;
148         fb_writedata = 1;

```

```

149      repeat(2) @(posedge clk);
150
151      //fb_WE = 0;
152      fb_addr_y = 5'b00000;
153      fb_addr_x = 6'b000000;
154      fb_writedata = 0;
155      fb_WE = 0;
156      repeat(2) @(posedge clk);
157
158      fb_addr_y = 5'b00010;
159      fb_addr_x = 6'b000010;
160      fb_writedata = 0;
161      fb_WE = 0;
162      repeat(2) @(posedge clk);
163
164      repeat(2)
165      @(posedge clk);
166      assert (fb_readdata == 1'h1) begin
167          $display ("WriteManyReadMany TEST 3 part 1 :
168             ↳ PASSED");
169      end
170      else $error("WriteManyReadMany TEST 3 part 1 : FAILED
171             ↳ (Got %h, Expected 1)", fb_readdata);
172
173      fb_addr_y = 5'b00000;
174      fb_addr_x = 6'b000000;
175      fb_writedata = 0;
176      fb_WE = 0;
177      repeat(2) @(posedge clk);
178
179      fb_addr_y = 5'b00100;
180      fb_addr_x = 6'b000100;
181      fb_writedata = 0;
182      fb_WE = 0;
183      repeat(2)

```

```

184     @(posedge clk);
185     assert (fb_readdata == 1'h1) begin
186         $display ("WriteManyReadMany TEST 3 part 2 :
187             ↳ PASSED");
188     end
189     else $error("WriteManyReadMany TEST 3 part 2 : FAILED
190             ↳ (Got %h, Expected 1)", fb_readdata);
191
192     fb_addr_y = 5'b00000;
193     fb_addr_x = 6'b000000;
194     fb_writedata = 0;
195     fb_WE = 0;
196     repeat(2) @(posedge clk);
197
198     fb_addr_y = 5'b01000;
199     fb_addr_x = 6'b001000;
200     fb_writedata = 0;
201     fb_WE = 0;
202     repeat(2) @(posedge clk);
203
204     repeat(2)
205     @(posedge clk);
206     assert (fb_readdata == 1'h1) begin
207         $display ("WriteManyReadMany TEST 3 part 3 :
208             ↳ PASSED");
209     end
210     else $error("WriteManyReadMany TEST 3 part 3 : FAILED
211             ↳ (Got %h, Expected 1)", fb_readdata);
212
213     fb_addr_y = 5'b00000;
214     fb_addr_x = 6'b000000;
215     fb_writedata = 0;
216     fb_WE = 0;
217     repeat(2) @(posedge clk);
218
219     fb_addr_y = 5'b10000;
220     fb_addr_x = 6'b010000;

```

```

217     fb_writedata = 0;
218     fb_WE = 0;
219     repeat(2) @(posedge clk);
220
221     repeat(2)
222     @(posedge clk);
223     assert (fb_readdata == 1'h1) begin
224         $display ("WriteManyReadMany TEST 3 part 4 :
225             ↳ PASSED");
226     end
227     else $error("WriteManyReadMany TEST 3 part 4 : FAILED
228             ↳ (Got %h, Expected 1)", fb_readdata);
229
230     fb_addr_y = 5'b00000;
231     fb_addr_x = 6'b0000000;
232     fb_writedata = 0;
233     fb_WE = 0;
234     repeat(2) @(posedge clk);
235
236     fb_addr_y = 5'b10000;
237     fb_addr_x = 6'b1000000;
238     fb_writedata = 0;
239     fb_WE = 0;
240     repeat(2) @(posedge clk);
241
242     repeat(2)
243     @(posedge clk);
244     assert (fb_readdata == 1'h0) begin
245         $display ("WriteManyReadMany TEST 3 part 5 :
246             ↳ PASSED");
247         total = total + 1;
248     end
249     else $error("OR TEST 3 part 5 : FAILED (Got %h,
250             ↳ Expected 0)", fb_readdata);
251
252     fb_addr_y = 6'h0000;
253     fb_addr_x = 5'h0000;

```

```

250     fb_writedata = 1'h0;
251     fb_WE = 1'h0;
252
253     reset = 1'h1;
254     repeat(2) @(posedge clk);
255     reset = 1'h0;
256     repeat(2) @(posedge clk);
257     //testReset(clk, fb_addr_y, fb_addr_x, fb_writedata,
258     //            → fb_WE, reset);
259   endtask
260
261 module fb_testbench();
262   logic clk;
263   logic reset;
264   logic [4:0] fb_addr_y;
265   logic [5:0] fb_addr_x;
266   logic fb_writedata;
267   logic fb_WE;
268   logic fb_readdata;
269   logic [7:0] VGA_R, VGA_G, VGA_B;
270   logic VGA_CLK, VGA_HS, VGA_VS, VGA_BLANK_n;
271   logic VGA_SYNC_n;
272   int total;
273
274   Chip8_framebuffer dut(.*);
275
276   initial begin
277     clk = 0;
278     reset = 0;
279     fb_addr_y = 5'b00000;
280     fb_addr_x = 6'b000000;
281     fb_writedata = 0;
282     fb_WE = 0;
283     total = 0;
284     forever
285       #20ns clk = ~clk;
286   end

```

```

287    initial begin
288
289        $display("Starting test script...");  

290        testWriteOne(clk, fb_addr_y, fb_addr_x, fb_writedata,  

291                      ↳ fb_WE, fb_readdata, reset, total);  

292        testWriteOneReadElse(clk, fb_addr_y, fb_addr_x,  

293                      ↳ fb_writedata, fb_WE, fb_readdata, reset, total);  

294        testWriteManyReadMany(clk, fb_addr_y, fb_addr_x,  

295                      ↳ fb_writedata, fb_WE, fb_readdata, reset, total);
296
297        repeat(2) @(posedge clk);
298
299    end
300
301 endmodule

```

### 7.2.8 alu\_testbench.sv

```

1  ****
2  * alu_testbench.sv
3  *
4  * Contains tests for the following instructions:
5  *      - OR          - bitwise OR
6  *      - AND         - bitwise AND
7  *      - XOR         - bitwise XOR
8  *      - ADD          - Addition
9  *      - MINUS        - Subtract
10 *      - LSHIFT       - Shift left
11 *      - RSHIFT       - Shift right
12 *      - EQUALS        - Equals compare
13 *      - GREATER       - Greater than compare
14 *      - INC          - Increment
15 *
16 * This module is solely used by the Chip8_CPU module, &
17 * relies on the ALU_f
18 * enum defined in enums.svh

```

```

18 *
19 * AUTHORS: David Watkins, Gabrielle Taylor
20 * Dependencies:
21 *   - enums.svh
22 *   - Chip8_CPU/Chip8_ALU.sv
23
24 -->
25 'include "enums.svh"
26
27 task automatic testReset(logic [15:0] input1, input2,
28                         ALU_f alu_op);
29     input1 = 16'h0000;
30     input2 = 16'h0000;
31     alu_op = ALU_f_NOP;
32 endtask
33
34 /**
35 * Tests the OR instruction for the ALU
36 *
37 * @test 1
38 * @input input1 = 16'hF5A0
39 * @input input2 = 16'hFA50
40 * @input alu_op = ALU_f_OR
41 * @expected result = 16'FFFO
42 * @expected alu_carry = 1'b0
43 */
44
45 task automatic testOR(ref logic clk, alu_carry,
46                       ref logic [15:0] input1, input2, result,
47                       ref ALU_f alu_op,
48                       ref int total);
49 //Setup test 1
50 repeat(2)
51     @(posedge clk);
52     input1 = 16'hF5A0;
53     input2 = 16'hFA50;
54     alu_op = ALU_f_OR;

```

```

55
56     repeat(2)
57         @(posedge clk);
58         assert (result == 16'hFFF0 && alu_carry == 1'b0) begin
59             $display ("OR TEST 1 : PASSED");
60             total = total + 1;
61         end
62         else $error("OR TEST 1 : FAILED (Got %h, Expected fff0)", 
63             result);
64
65         testReset(input1, input2, alu_op);
66
67
68 /**
69 * Tests the AND instruction for the ALU
70 *
71 * @test 1
72 * @input input1 = 16'hF5A0
73 * @input input2 = 16'hFA50
74 * @input alu_op = ALU_f_AND
75 * @expected result = 16'F000
76 * @expected alu_carry = 1'b0
77 */
78 task automatic testAND(ref logic clk, alu_carry,
79                         ref logic [15:0] input1, input2, result,
80                         ref ALU_f alu_op,
81                         ref int total);
82     //Setup test 1
83     repeat(2)
84         @(posedge clk);
85         input1 = 16'hF5A0;
86         input2 = 16'hFA50;
87         alu_op = ALU_f_AND;
88
89     repeat(2)
90         @(posedge clk);

```

```

91      assert (result == 16'hF000 && alu_carry == 1'b0) begin
92          $display ("AND TEST 1 : PASSED");
93          total = total + 1;
94      end
95      else $error("AND TEST 1 : FAILED (Got %h, Expected f000)",
96                  result);
97
98      testReset(input1, input2, alu_op);
99
100
101 /**
102 * Tests the XOR instruction for the ALU
103 *
104 * @test 1
105 * @input input1 = 16'hF5A0
106 * @input input2 = 16'hFA50
107 * @input alu_op = ALU_f_XOR
108 * @expected result = 16'h0FF0
109 * @expected alu_carry = 1'b0
110 */
111 task automatic testXOR(ref logic clk, alu_carry,
112                         ref logic [15:0] input1, input2, result,
113                         ref ALU_f alu_op,
114                         ref int total);
115     //Setup test 1
116     repeat(2)
117         @(posedge clk);
118         input1 = 16'hF5A0;
119         input2 = 16'hFA50;
120         alu_op = ALU_f_XOR;
121
122     repeat(2)
123         @(posedge clk);
124     assert (result == 16'h0FF0 && alu_carry == 1'b0) begin
125         $display ("XOR TEST 1 : PASSED");
126         total = total + 1;

```

```

127     end
128     else $error("XOR TEST 1 : FAILED (Got %h, Expected Off0)", 
129                  result);
130
131     testReset(input1, input2, alu_op);
132
133
134 /**
135 * Tests the MINUS instruction for the ALU
136 *
137 * @test 1
138 * @input input1 = 16'd180
139 * @input input2 = 16'd180
140 * @input alu_op = ALU_f_ADD
141 * @expected result = 16'd360
142 * @expected alu_carry = 1'b1
143 *
144 * @test 2
145 * @input input1 = 16'd5
146 * @input input2 = 16'd5
147 * @input alu_op = ALU_f_ADD
148 * @expected result = 16'd10
149 * @expected alu_carry = 1'b1
150 */
151 task automatic testADD(ref logic clk, alu_carry,
152                         ref logic [15:0] input1, input2, result,
153                         ref ALU_f alu_op,
154                         ref int total);
155 //Setup test 1
156 repeat(2)
157     @(posedge clk);
158     input1 = 16'd180;
159     input2 = 16'd180;
160     alu_op = ALU_f_ADD;
161
162     repeat(2)

```

```

163     @(posedge clk);
164     assert (result == 16'd360 && alu_carry == 1'b1) begin
165         $display ("ADD TEST 1 : PASSED");
166         total = total + 1;
167     end
168     else $error("ADD TEST 1 : FAILED (Got %d, Expected 360)
169             → (Got %d, Expected 1)", result, alu_carry);

170     //Setup test 2
171     repeat(2)
172         @(posedge clk);
173         input1 = 16'd5;
174         input2 = 16'd5;
175         alu_op = ALU_f_ADD;
176
177     repeat(2)
178         @(posedge clk);
179     assert (result == 16'd10 && alu_carry == 1'b0) begin
180         $display ("ADD TEST 2 : PASSED");
181         total = total + 1;
182     end
183     else $error("ADD TEST 2 : FAILED (Got %d, Expected 10)
184             → (Got %d, Expected 0)", result, alu_carry);

185     testReset(input1, input2, alu_op);
186 endtask
187
188
189 /**
190 * Tests the MINUS instruction for the ALU
191 *
192 * @test 1
193 * @input input1 = 16'hC3C3
194 * @input input2 = 16'hC3C3
195 * @input alu_op = ALU_f_MINUS
196 * @expected result = 16'h0000
197 * @expected alu_carry = 1'b0

```

```

198  *
199  * @test 2
200  * @input input1 = 16'hE0A5
201  * @input input2 = 16'h7003
202  * @input alu_op = ALU_f_MINUS
203  * @expected result = 16'h70A2
204  * @expected alu_carry = 1'b0
205  *
206  * @test 3
207  * @input input1 = 16'h7003
208  * @input input2 = 16'hE0A5
209  * @input alu_op = ALU_f_MINUS
210  * @expected result = 16'd36702
211  * @expected alu_carry = 1'b0
212 */
213 task automatic testMINUS(ref logic clk, alu_carry,
214                         ref logic [15:0] input1, input2, result,
215                         ref ALU_f alu_op,
216                         ref int total);
217
218 //Setup test 1
219 repeat(2)
220     @(posedge clk);
221     input1 = 16'hC3C3;
222     input2 = 16'hC3C3;
223     alu_op = ALU_f_MINUS;
224
225 repeat(2)
226     @(posedge clk);
227     assert (result == 16'h0000 && alu_carry == 1'b0) begin
228         $display ("MINUS TEST 1 : PASSED");
229         total = total + 1;
230     end
231     else $error("MINUS TEST 1 : FAILED (Got %h, Expected 0000
232             ↳ (Got %d, Expected 1)", result, alu_carry);
233
234 //Setup test 2

```

```

234     repeat(2)
235         @(posedge clk);
236         input1 = 16'hE0A5;
237         input2 = 16'h7003;
238         alu_op = ALU_f_MINUS;
239
240     repeat(2)
241         @(posedge clk);
242         assert (result == 16'h70A2 && alu_carry == 1'b0) begin
243             $display ("MINUS TEST 2 : PASSED");
244             total = total + 1;
245         end
246         else $error("MINUS TEST 2 : FAILED (Got %h, Expected 70a2)
247             ↪ (Got %d, Expected 1)", result, alu_carry);
248
249 //Setup test 3
250 repeat(2)
251     @(posedge clk);
252     input1 = 16'h7003;
253     input2 = 16'hE0A5;
254     alu_op = ALU_f_MINUS;
255
256 repeat(2)
257     @(posedge clk);
258     assert (result == 16'd36702 && alu_carry == 1'b1) begin
259         $display ("MINUS TEST 3 : PASSED");
260         total = total + 1;
261     end
262     else $error("MINUS TEST 3 : FAILED (Got %d, Expected
263             ↪ 36702) (Got %d, Expected 1)", result, alu_carry);
264
265     testReset(input1, input2, alu_op);
266
267 endtask
268
269 /**
270 * Tests the lSHIFT instruction for the ALU

```

```

269 *
270 * @test 1
271 * @input input1 = 16'h0031
272 * @input input2 = 16'h0002
273 * @input alu_op = ALU_f_LSHIFT
274 * @expected result = 16'h00C4
275 * @expected alu_carry = 1'b0
276 *
277 * @test 2
278 * @input input1 = 16'h1111
279 * @input input2 = 16'h0001
280 * @input alu_op = ALU_f_LSHIFT
281 * @expected result = 16'h2222
282 * @expected alu_carry = 1'b0
283 */
284 task automatic testLSHIFT(ref logic clk, alu_carry,
285                         ref logic [15:0] input1, input2, result,
286                         ref ALU_f alu_op,
287                         ref int total);
288 //Setup test 1
289 repeat(2)
290     @(posedge clk);
291     input1 = 16'h0031;
292     input2 = 16'h0002;
293     alu_op = ALU_f_LSHIFT;
294
295 repeat(2)
296     @(posedge clk);
297     assert (result == 16'h00C4 && alu_carry == 1'b0) begin
298         $display ("LSHIFT TEST 1 : PASSED");
299         total = total + 1;
300     end
301     else $error("LSHIFT TEST 1 : FAILED (Got %h, Expected
302             → 00c4)", result);
303 //Setup test 2
304 repeat(2)

```

```

305      @(posedge clk);
306      input1 = 16'h1111;
307      input2 = 16'h0001;
308      alu_op = ALU_f_LSHIFT;
309
310      repeat(2)
311          @(posedge clk);
312          assert (result == 16'h2222 && alu_carry == 1'b0) begin
313              $display ("LSHIFT TEST 2 : PASSED");
314              total = total + 1;
315          end
316          else $error("LSHIFT TEST 2 : FAILED (Got %h, Expected
317              ↳ 2222)", result);
318
319      testReset(input1, input2, alu_op);
320
321
322 /**
323 * Tests the RSHIFT instruction for the ALU
324 *
325 * @test 1
326 * @input input1 = 16'h0031
327 * @input input2 = 16'h0002
328 * @input alu_op = ALU_f_RSHIFT
329 * @expected result = 16'h000C
330 * @expected alu_carry = 1'b0
331 *
332 * @test 2
333 * @input input1 = 16'h1111
334 * @input input2 = 16'h0001
335 * @input alu_op = ALU_f_RSHIFT
336 * @expected result = 16'h0888
337 * @expected alu_carry = 1'b0
338 */
339 task automatic testRSHIFT(ref logic clk, alu_carry,
340                         ref logic [15:0] input1, input2, result,

```

```

341           ref ALU_f alu_op,
342           ref int total);
343 //Setup test 1
344 repeat(2)
345     @(posedge clk);
346     input1 = 16'h0031;
347     input2 = 16'h0002;
348     alu_op = ALU_f_RSHIFT;
349
350 repeat(2)
351     @(posedge clk);
352 assert (result == 16'h000C && alu_carry == 1'b0) begin
353     $display ("RSHIFT TEST 1 : PASSED");
354     total = total + 1;
355 end
356 else $error("RSHIFT TEST 1 : FAILED (Got %h, Expected
357             ← 000c)", result);
358
359 //Setup test 2
360 repeat(2)
361     @(posedge clk);
362     input1 = 16'h1111;
363     input2 = 16'h0001;
364     alu_op = ALU_f_RSHIFT;
365
366 repeat(2)
367     @(posedge clk);
368 assert (result == 16'h0888 && alu_carry == 1'b0) begin
369     $display ("RSHIFT TEST 2 : PASSED");
370     total = total + 1;
371 end
372 else $error("RSHIFT TEST 2 : FAILED (Got %h, Expected
373             ← 0888)", result);
374
375 testReset(input1, input2, alu_op);
376 endtask

```

```

376
377 /**
378 * Tests the GREATER instruction for the ALU
379 *
380 * @test 1
381 * @input input1 = 16'd180
382 * @input input2 = 16'd15
383 * @input alu_op = ALU_f_GREATER
384 * @expected result = 16'd1
385 * @expected alu_carry = 1'b0
386 *
387 * @test 2
388 * @input input1 = 16'd15
389 * @input input2 = 16'd180
390 * @input alu_op = ALU_f_GREATER
391 * @expected result = 16'd0
392 * @expected alu_carry = 1'b0
393 *
394 * @test 3
395 * @input input1 = 16'd15
396 * @input input2 = 16'd15
397 * @input alu_op = ALU_f_GREATER
398 * @expected result = 16'd0
399 * @expected alu_carry = 1'b0
400 */
401 task automatic testGREATER(ref logic clk, alu_carry,
402                             ref logic [15:0] input1, input2, result,
403                             ref ALU_f alu_op,
404                             ref int total);
405 //Setup test 1
406 repeat(2)
407     @(posedge clk);
408     input1 = 16'd180;
409     input2 = 16'd15;
410     alu_op = ALU_f_GREATER;
411
412 repeat(2)
413     @(posedge clk);

```

```

414     assert (result == 16'd1 && alu_carry == 1'b0) begin
415         $display ("GREATER TEST 1 : PASSED");
416         total = total + 1;
417     end
418     else $error("GREATER TEST 1 : FAILED (Got %d, Expected
419             → 1)", result);
420
421 //Setup test 2
422 repeat(2)
423     @(posedge clk);
424     input1 = 16'd15;
425     input2 = 16'd180;
426     alu_op = ALU_f_GREATER;
427
428 repeat(2)
429     @(posedge clk);
430     assert (result == 16'd0 && alu_carry == 1'b0) begin
431         $display ("GREATER TEST 2 : PASSED");
432         total = total + 1;
433     end
434     else $error("GREATER TEST 2 : FAILED (Got %d, Expected
435             → 0)", result);
436
437 //Setup test 3
438 repeat(2)
439     @(posedge clk);
440     input1 = 16'd15;
441     input2 = 16'd15;
442     alu_op = ALU_f_GREATER;
443
444 repeat(2)
445     @(posedge clk);
446     assert (result == 16'd0 && alu_carry == 1'b0) begin
447         $display ("GREATER TEST 3 : PASSED");
448         total = total + 1;
449     end

```

```

448     else $error("GREATER TEST 3 : FAILED (Got %d, Expected
449      ↪  0)", result);
450
451     testReset(input1, input2, alu_op);
452
453
454 /**
455 * Tests the EQUALS instruction for the ALU
456 *
457 * @test 1
458 * @input input1 = 16'd8
459 * @input input2 = 16'd8
460 * @input alu_op = ALU_f_EQUALS
461 * @expected result = 16'd1
462 * @expected alu_carry = 1'b0
463 *
464 * @test 2
465 * @input input1 = 16'd8
466 * @input input2 = 16'd9
467 * @input alu_op = ALU_f_EQUALS
468 * @expected result = 16'd0
469 * @expected alu_carry = 1'b0
470 */
471 task automatic testEQUALS(ref logic clk, alu_carry,
472                         ref logic [15:0] input1, input2, result,
473                         ref ALU_f alu_op,
474                         ref int total);
475 //Setup test 1
476 repeat(2)
477   @(posedge clk);
478   input1 = 16'd8;
479   input2 = 16'd8;
480   alu_op = ALU_f_EQUALS;
481
482 repeat(2)
483   @(posedge clk);

```

```

484     assert (result == 16'd1 && alu_carry == 1'b0) begin
485         $display ("EQUALS TEST 1 : PASSED");
486         total = total + 1;
487     end
488     else $error("EQUALS TEST 1 : FAILED (Got %d, Expected 1)",
489                 result);
490
491     //Setup test 2
492     repeat(2)
493         @(posedge clk);
494         input1 = 16'd8;
495         input2 = 16'd9;
496         alu_op = ALU_f_EQUALS;
497
498     repeat(2)
499         @(posedge clk);
500     assert (result == 16'd0 && alu_carry == 1'b0) begin
501         $display ("EQUALS TEST 2 : PASSED");
502         total = total + 1;
503     end
504     else $error("EQUALS TEST 2 : FAILED (Got %d, Expected 0)",
505                 result);
506
507     testReset(input1, input2, alu_op);
508 endtask
509
510 /**
511 * Tests the INC instruction for the ALU
512 *
513 * @test 1
514 * @input input1 = 16'd8
515 * @input alu_op = ALU_f_INC
516 * @expected result = 16'd26
517 * @expected alu_carry = 1'b0
518 */
519 task automatic testINC(ref logic clk, alu_carry,

```

```

519          ref logic [15:0] input1, input2, result,
520          ref ALU_f alu_op,
521          ref int total);
522
//Setup
523 repeat(2)
524     @(posedge clk);
525     input1 = 16'd8;
526     input2 = 16'h0000;
527     alu_op = ALU_f_INC;
528     repeat(16) begin
529         @(posedge clk);
530         input1 = result;
531     end
532
//Check
533 repeat(2)
534     @(posedge clk);
535     assert (result == 16'd25 && alu_carry == 1'b0) begin
536         $display ("INC TEST : PASSED");
537         total = total + 1;
538     end
539     else $error("INC TEST : FAILED (Got %d, Expected 24)",
540      ↪ result);
541
542     testReset(input1, input2, alu_op);
543 endtask
544
545
546 module alu_testbench();
547     logic clk;
548     logic alu_carry;
549     logic [15:0] result;
550     logic[15:0] input1, input2;
551     ALU_f alu_op;
552     int total;
553
554     Chip8_ALU dut(

```

```

555     .input1(input1),
556     .input2(input2),
557     .sel(alu_op),
558     .out(result),
559     .alu_carry(alu_carry));
560
561 initial begin
562     clk = 0;
563     input1 = 16'h0000;
564     input2 = 16'h0000;
565     alu_op = ALU_f_NOP;
566     forever
567         #20ns clk = ~clk;
568 end
569
570 initial begin
571     $display("Starting test script...\"");
572     testOR(clk, alu_carry, input1, input2, result, alu_op,
573             ↳ total);
574     testAND(clk, alu_carry, input1, input2, result,
575             ↳ alu_op, total);
576     testXOR(clk, alu_carry, input1, input2, result,
577             ↳ alu_op, total);
578     testADD(clk, alu_carry, input1, input2, result,
579             ↳ alu_op, total);
580     testMINUS(clk, alu_carry, input1, input2, result,
581             ↳ alu_op, total);
582     testLSHIFT(clk, alu_carry, input1, input2, result,
583             ↳ alu_op, total);
584     testRSHIFT(clk, alu_carry, input1, input2, result,
585             ↳ alu_op, total);
586     testGREATER(clk, alu_carry, input1, input2, result,
587             ↳ alu_op, total);
588     testEQUALS(clk, alu_carry, input1, input2, result,
589             ↳ alu_op, total);
590     testINC(clk, alu_carry, input1, input2, result,
591             ↳ alu_op, total);
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850

```

```

583         $display("TESTS PASSED : %d", total);
584     end
585
586 endmodule
587

```

## 7.3 Linux Code

### 7.3.1 chip8.c

```

1  /*
2  * Userspace program that communicates with the vga_ball
3  * device driver
4  * primarily through ioctls
5  *
6  * David Watkins (djw2146), Ashley Kling (ask2203)
7  * Columbia University
8  */
9
10 #include <stdio.h>
11 #include "chip8driver.h"
12 #include <sys/ioctl.h>
13 #include <sys/types.h>
14 #include <sys/stat.h>
15 #include <fcntl.h>
16 #include <string.h>
17 #include <unistd.h>
18 #include <signal.h>
19 #include <stdlib.h>
20 #include <pthread.h>
21
22 #include "usbkeyboard.h"
23 static int CHIP8_FONTSET[] =
24 {
25     0xF0, 0x90, 0x90, 0x90, 0xF0, //0
26     0x20, 0x60, 0x20, 0x20, 0x70, //1

```

```

27         0xF0, 0x10, 0xF0, 0x80, 0xF0, //2
28         0xF0, 0x10, 0xF0, 0x10, 0xF0, //3
29         0x90, 0x90, 0xF0, 0x10, 0x10, //4
30         0xF0, 0x80, 0xF0, 0x10, 0xF0, //5
31         0xF0, 0x80, 0xF0, 0x90, 0xF0, //6
32         0xF0, 0x10, 0x20, 0x40, 0x40, //7
33         0xF0, 0x90, 0xF0, 0x90, 0xF0, //8
34         0xF0, 0x90, 0xF0, 0x10, 0xF0, //9
35         0xF0, 0x90, 0xF0, 0x90, 0x90, //A
36         0xE0, 0x90, 0xE0, 0x90, 0xE0, //B
37         0xF0, 0x80, 0x80, 0x80, 0xF0, //C
38         0xE0, 0x90, 0x90, 0x90, 0xE0, //D
39         0xF0, 0x80, 0xF0, 0x80, 0xF0, //E
40         0xF0, 0x80, 0xF0, 0x80, 0x80 //F
41     };
42
43 #define FONTSET_LENGTH 80
44 #define MEMORY_START 0x200
45 #define MEMORY_END 0x1000
46
47 int chip8_fd;
48 struct libusb_device_handle *keyboard;
49 uint8_t endpoint_address;
50 FILE *fp;
51
52 void quit_program(int signal) {
53     printf("Chip8 is terminating\n");
54     close(chip8_fd);
55     exit(0);
56 }
57
58 void chip8_write(chip8_opcode *op) {
59     if(ioctl(chip8_fd, CHIP8_WRITE_ATTR, op)) {
60         perror("ioctl(CHIP8_WRITE_ATTR) failed");
61         quit_program(0);
62     }
63 }
64

```

```

65 void chip8_read(chip8_opcode *op) {
66     if(ioctl(chip8_fd, CHIP8_READ_ATTR, op)) {
67         perror("ioctl(CHIP8_READ_ATTR) failed");
68         printf("(%d, %d)\n", op->addr, op->data);
69         quit_program(0);
70     }
71 }
72
73 void setFramebuffer(int x, int y, int value) {
74     chip8_opcode op;
75     op.addr = FRAMEBUFFER_ADDR;
76     op.data = (1 << 12) | ((value & 0x1) << 11) | ((x & 0x3f)
77         << 5) | (y & 0x1f);
78     chip8_write(&op);
79 }
80
81 int readFramebuffer(int x, int y) {
82     chip8_opcode op;
83     op.addr = FRAMEBUFFER_ADDR;
84     op.data = (0 << 12) | (0 << 11) | ((x & 0x3f) << 5) | (y &
85         0x1f);
86     chip8_read(&op);
87     return op.readdata;
88 }
89
90 void flipPixel(int x, int y) {
91     int px = readFramebuffer(x, y);
92     setFramebuffer(x, y, !px);
93 }
94
95 void setMemory(int address, int data) {
96     chip8_opcode op;
97     op.addr = MEMORY_ADDR;
98     op.data = (1 << 20) | ((address & 0xffff) << 8) | (data &
99         0xff);
100    chip8_write(&op);
101 }
```

```

100 int readMemory(int address) {
101     chip8_opcode op;
102     op.addr = MEMORY_ADDR;
103     op.data = (0 << 20) | ((address & 0xffff) << 8) | (0 &
104         ~ 0xff);
105     chip8_read(&op);
106     return (op.readdata & 0xff);
107 }
108 void setIRegister(int data) {
109     chip8_opcode op;
110     op.addr = I_ADDR;
111     op.data = (data & 0xffff);
112     chip8_write(&op);
113 }
114
115 int readIRegister() {
116     chip8_opcode op;
117     op.addr = I_ADDR;
118     chip8_read(&op);
119     return op.readdata;
120 }
121
122 int readRegister(int reg) {
123     chip8_opcode op;
124     op.addr = V0_ADDR + 4 * (reg & 0xf);
125     chip8_read(&op);
126     return op.readdata;
127 }
128
129 void writeRegister(int reg, int value) {
130     chip8_opcode op;
131     op.addr = V0_ADDR + 4 * (reg & 0xf);
132     op.data = value & 0xff;
133     chip8_write(&op);
134 }
135
136 /*

```

```

137 * Load the font set onto the chip8 sequentially
138 * Uses the op codes specified in chip8driver.h
139 */
140 void loadfontset() {
141     int i;
142     for(i = 0; i < FONTSET_LENGTH; ++i) {
143         setMemory(i, CHIP8_FONTSET[i]);
144         // int mem_val = readMemory(i);
145         // printf("(Address: %d) Wrote: %d, Read: %d\n", i,
146             // CHIP8_FONTSET[i], mem_val);
147         int got = readMemory(i);
148         if (CHIP8_FONTSET[i] != got) {
149             printf("Memory mismatch (expected: %d, got:
150                 %d)\n", CHIP8_FONTSET[i], got);
151         }
152     }
153 }
154 void refreshFrameBuffer() {
155     int x, y;
156     for(x = 0; x < 64; ++x) {
157         for(y = 0; y < 32; ++y) {
158             // int mem_val = readFramebuffer(x, y);
159             setFramebuffer(x, y, 0);
160
161             // printf("(x: %d, y: %d) Wrote: %d, Read: %d\n",
162                 // x, y, !mem_val, mem_val);
163         }
164     }
165 */
166 * Load a ROM file byte by byte onto the chip8
167 * Uses the op codes specified in chip8driver.h
168 */
169 void loadROM(const char* romfilename) {
170     FILE *romfile;

```

```

172     char buffer;
173     long filelen;
174     int i;
175
176     romfile = fopen(romfilename, "rb");
177     fseek(romfile, 0, SEEK_END);
178     filelen = ftell(romfile);
179     rewind(romfile);
180
181     for(i = 0; i < filelen && i < MEMORY_END - MEMORY_START;
182         i++) {
183         fread(&buffer, 1, 1, romfile);
184
185         setMemory(MEMORY_START + i, buffer);
186         int got = readMemory(MEMORY_START + i);
187         if (buffer != got) {
188             printf("Memory mismatch (expected: %d, got:
189                 %d)\n", buffer, got);
190         }
191     }
192
193     for(i = i; i < MEMORY_END - MEMORY_START; ++i) {
194         setMemory(MEMORY_START + i, 0);
195     }
196
197     fclose(romfile); // Close the file
198 }
199
200 void resetMemory() {
201     int i;
202     for(i = 0; i < MEMORY_END; i++) {
203         setMemory(i, 0);
204     }
205 }
206
207 void startChip8() {
208     chip8_opcode op;
209     op.addr = STATE_ADDR;

```

```

208     op.data = RUNNING_STATE;
209
210     chip8_write(&op);
211 }
212
213 void pauseChip8() {
214     chip8_opcode op;
215     op.addr = STATE_ADDR;
216     op.data = PAUSED_STATE;
217
218     chip8_write(&op);
219 }
220
221 void runInstructionChip8() {
222     chip8_opcode op;
223     op.addr = STATE_ADDR;
224     op.data = RUN_INSTRUCTION_STATE;
225
226     chip8_write(&op);
227 }
228
229 int chip8isRunning() {
230     chip8_opcode op;
231     op.addr = STATE_ADDR;
232     chip8_read(&op);
233     return op.readdata == RUNNING_STATE;
234 }
235
236 int chip8isPaused() {
237     chip8_opcode op;
238     op.addr = STATE_ADDR;
239     chip8_read(&op);
240     return op.readdata == PAUSED_STATE;
241 }
242
243 int chip8isRunInstruction() {
244     chip8_opcode op;
245     op.addr = STATE_ADDR;

```

```

246     chip8_read(&op);
247     return op.readdata == RUN_INSTRUCTION_STATE;
248 }
249
250 int readPC() {
251     chip8_opcode op;
252     op.addr = PROGRAM_COUNTER_ADDR;
253     chip8_read(&op);
254     // fprintf(fp, "Instruction: %04x, PC: %d\n",
255     // → (op.readdata & 0xfffff000) >> 12, (op.readdata &
256     // → 0xffff));
257     return (op.readdata & 0xffff);
258 }
259
260 void writePC(int pc) {
261     chip8_opcode op;
262     op.addr = PROGRAM_COUNTER_ADDR;
263     op.data = pc;
264     chip8_write(&op);
265 }
266
267 void printMemory() {
268     int i = 0;
269     for(i = 0; i < MEMORY_END; ++i) {
270         printf("%d ", readMemory(i));
271     }
272     printf("\n");
273 }
274
275 void resetStack() {
276     chip8_opcode op;
277     op.addr = STACK_ADDR;
278     chip8_write(&op);
279 }
280
281 int readSoundTimer() {
282     chip8_opcode op;
283     op.addr = SOUND_TIMER_ADDR;

```

```

282     chip8_read(&op);
283     return op.readdata;
284 }
285
286 void writeSoundTimer(int value) {
287     chip8_opcode op;
288     op.addr = SOUND_TIMER_ADDR;
289     op.data = value;
290     chip8_write(&op);
291 }
292
293 int readDelayTimer() {
294     chip8_opcode op;
295     op.addr = DELAY_TIMER_ADDR;
296     chip8_read(&op);
297     return op.readdata;
298 }
299
300 void writeDelayTimer(int value) {
301     chip8_opcode op;
302     op.addr = DELAY_TIMER_ADDR;
303     op.data = value;
304     chip8_write(&op);
305 }
306
307 void writeInstruction(int instruction) {
308     chip8_opcode op;
309     op.addr = INSTRUCTION_ADDR;
310     op.data = instruction;
311     chip8_write(&op);
312
313     usleep(1000000);
314 }
315
316 int readInstruction() {
317     chip8_opcode op;
318     op.addr = INSTRUCTION_ADDR;
319     chip8_read(&op);

```

```

320     return op.readdata;
321 }
322
323
324 void chip8writekeypress(char val, unsigned int ispressed) {
325     chip8_opcode op;
326     op.addr = KEY_PRESS_ADDR;
327     op.data = ((ispressed & 0x1) << 4) | (val & 0xf);
328     chip8_write(&op);
329 }
330
331 void printKeyState() {
332     chip8_opcode op;
333     op.addr = KEY_PRESS_ADDR;
334     chip8_read(&op);
335
336     printf("Is pressed: %d, Key val: %d, raw value: %d\n",
337           → (op.readdata & 0x10) >> 4, (op.readdata & 0xf),
338           → op.readdata);
339 }
340
341 void writeReset() {
342     chip8_opcode op;
343     op.addr = RESET_ADDR;
344     chip8_write(&op);
345 }
346
347 void printStatus(FILE *out, int index) {
348
349     fprintf(out, "Status %d\n", index);
350     if(chip8isPaused()) {
351         fprintf(out, "Paused\n");
352     } else if(chip8isRunning()) {
353         fprintf(out, "Running\n");
354     } else {
355         fprintf(out, "Run Instruction\n");
356     }
357     int pc = readPC();

```

```

356     int mem = readMemory(pc);
357     int mem2 = readMemory(pc + 1);
358     fprintf(out, "Program counter is: %d, instruction is: %04x
359             / %04x\n", pc, mem << 4 | mem2, readInstruction());
360     fprintf(out, "I register: %d\n", readIRegister());
361     int i;
362     for(i = 0; i < 0x10; ++i) {
363         fprintf(out, "v%d: %d\n", i, readRegister(i));
364     }
365
366     fprintf(out, "Sound timer: %d\n", readSoundTimer());
367     fprintf(out, "Delay timer: %d\n\n", readDelayTimer());
368 }
369
370 void resetChip8(const char* filename) {
371     //Need to write to registers and all
372     //Reload font set etc.
373     pauseChip8();
374     resetMemory();
375
376     loadfontset();
377     if(filename != 0)
378         loadROM(filename);
379     refreshFrameBuffer();
380
381     int i;
382     for(i = 0; i < 0x10; ++i) {
383         writeRegister(i, 0);
384     }
385
386     // printMemory();
387     writePC(0x200);
388     setIRegister(0);
389     resetStack();
390     chip8writekeypress(0, 0);
391     writeSoundTimer(0);
392     writeDelayTimer(0);

```

```

393     printStatus(stdout, 0);
394 }
395
396 /*
397 * Checks to see if a key is pressed, or depressed
398 * Then writes the associated action to the chip8 device
399 */
400 void checkforkeypress(const char *file) {
401     struct usb_keyboard_packet packet;
402     int transferred;
403     char keystate[12];
404
405     libusb_interrupt_transfer(keyboard, endpoint_address,
406         → (unsigned char *) &packet, sizeof(packet),
407         → &transferred, 0);
408     if (transferred == sizeof(packet)) {
409         sprintf(keystate, "%02x %02x %02x", packet.modifiers,
410             → packet.keycode[0], packet.keycode[1]);
411         char val[1];
412         if (kbiskeypad(&packet, val)) {
413             chip8writekeypress(val[0], 1);
414         } else if(kbisstart(&packet)) {
415             startChip8();
416         } else if(kbispause(&packet)) {
417             pauseChip8();
418         } else if(kbisreset(&packet)) {
419             resetChip8(file);
420         } else {
421             chip8writekeypress(0, 0);
422         }
423     } else {
424         printf("Size mismatch %d %d\n", sizeof(packet),
425             → transferred);
426     }
427 }
428
429 void *status_thread_f(void *ignored)
430 {

```

```

427     int index = 0;
428     while(1) {
429         printStatus(fp, index++);
430         usleep(4000);
431     }
432
433     return NULL;
434 }
435
436 int main(int argc, char** argv)
437 {
438     int runType = 0;
439     if(argc != 2 && argc != 3) {
440         printf("Usage: chip8 <romfilename>\n");
441         exit(1);
442     }
443
444     if(argc == 3) runType = 1;
445
446     /* Open the keyboard */
447     if ( (keyboard = openkeyboard(&endpoint_address)) == NULL
448         ) {
449         fprintf(stderr, "Did not find a keyboard\n");
450         exit(1);
451     }
452
453     static const char filename[] = "/dev/vga_led";
454     if ( (chip8_fd = open(filename, O_RDWR)) == -1) {
455         fprintf(stderr, "could not open %s\n", filename);
456         return -1;
457     }
458
459     signal(SIGINT, quit_program);
460
461     fp = fopen("log.txt", "w+");
462
463     pthread_t status_thread;

```

```

464     if(runType == 0) {
465         resetChip8(argv[1]);
466         // pthread_create(&status_thread, NULL,
467         //                 status_thread_f, NULL);
468
469         while(chip8isRunning() || chip8isPaused()) {
470             // printStatus(stdout, 0);
471             checkforkeypress(argv[1]);
472             printKeyState();
473         }
474
475         /* Terminate the status thread */
476         // pthread_cancel(status_thread);
477
478         /* Wait for the status thread to finish */
479         // pthread_join(status_thread, NULL);
480     }
481
482     fclose(fp);
483
484     printf("Chip8 is terminating\n");
485     close(chip8_fd);
486     return 0;
487 }
```

### 7.3.2 chip8driver.c

```

1  /*
2   * Device driver for the CHIP8 SystemVerilog Emulator
3   *
4   * A Platform device implemented using the misc subsystem
5   *
6   * Columbia University
7   *
8   * References:
9   * Linux source: Documentation/driver-model/platform.txt
10    drivers/misc/arm-charlcd.c
```

```

11  * http://www.linuxforu.com/tag/linux-device-drivers/
12  * http://free-electrons.com/docs/
13  *
14  * "make" to build
15  * insmod chip8driver.ko
16  *
17  * Check code style with
18  * checkpatch.pl --file --no-tree chip8driver.c
19  */
20
21 #include <linux/module.h>
22 #include <linux/init.h>
23 #include <linux/errno.h>
24 #include <linux/version.h>
25 #include <linux/kernel.h>
26 #include <linux/platform_device.h>
27 #include <linux/miscdevice.h>
28 #include <linux/slab.h>
29 #include <linux/io.h>
30 #include <linux/of.h>
31 #include <linux/of_address.h>
32 #include <linux/fs.h>
33 #include <linux/uaccess.h>
34 #include "chip8driver.h"
35
36 #define DRIVER_NAME "vga_led"
37
38 /*
39  * Information about our device
40  */
41 struct chip8_dev {
42     struct resource res;           /* Resource: our registers
43     ↳ */
44     void __iomem *virtbase;        /* Where registers can be
45     ↳ accessed in memory */
46 } dev;
47
48 /*

```

```

47 * Writes an opcode (defined in chip8driver.h) to the device
48 */
49 static void write_op(unsigned int addr, unsigned int
50    ↳ instruction) {
51     iowrite32(instruction, dev.virtbase + addr);
52 }
53 /*
54 * Reads a value after sending a proper write opcode to the
55    ↳ device
56 */
57 static int read_value(unsigned int addr) {
58     return ioread32(dev.virtbase + addr);
59 }
60 /*
61 * Checks to see if the address is validly formatted
62 */
63 static int isValidInstruction(unsigned int addr, unsigned int
64    ↳ instruction, int isWrite) {
65     switch(addr) {
66         //Register instructions are always okay
67         case V0_ADDR: return 1;
68         case V1_ADDR: return 1;
69         case V2_ADDR: return 1;
70         case V3_ADDR: return 1;
71         case V4_ADDR: return 1;
72         case V5_ADDR: return 1;
73         case V6_ADDR: return 1;
74         case V7_ADDR: return 1;
75         case V8_ADDR: return 1;
76         case V9_ADDR: return 1;
77         case VA_ADDR: return 1;
78         case VB_ADDR: return 1;
79         case VC_ADDR: return 1;
80         case VD_ADDR: return 1;
81         case VE_ADDR: return 1;
82         case VF_ADDR: return 1;

```

```

82     case I_ADDR:    return 1;
83
84     //Timer instructions are always okay
85     case SOUND_TIMER_ADDR: return 1;
86     case DELAY_TIMER_ADDR: return 1;
87
88     //Stack instructions are only valid if they conform
89     //to stack size
90     //Always looks at last three nibbles
91     case STACK_POINTER_ADDR: return !isWrite ||
92         (instruction >= 0 && instruction < 64);
93     case STACK_ADDR:          return 1;
94
95     //Handle state transition
96     case STATE_ADDR: switch(instruction) {
97         case RUNNING_STATE: return 1;
98         case RUN_INSTRUCTION_STATE: return 1;
99         case PAUSED_STATE: return 1;
100        default: return !isWrite;
101    }
102
103    //Memory address
104    case MEMORY_ADDR:
105        //0000_0000_0001_AAAA_AAAA_AAAA_DDDD_DDDD
106        if(isWrite) return 1;
107        else return 2;
108
109    //Program Counter will always look at the last 3
110    //nibbles
111    case PROGRAM_COUNTER_ADDR: return 1;
112
113    //Always considers last nibble
114    case KEY_PRESS_ADDR: return 1;
115
116    //Make sure X, Y, and data values conform
117    //Data value will always be 1 byte
118    case FRAMEBUFFER_ADDR:
119        //0000_0000_0000_0001_DXXX_XXYY_YYYY

```

```

117         if(isWrite) return 1;
118     else         return 2;
119
120     case INSTRUCTION_ADDR: return 1;
121     case RESET_ADDR : return 1;
122
123     default: break;
124 }
125
126     return 0;
127 }
128
129
130 /*
131 * Handle ioctl() calls from userspace:
132 * Read or write the segments on single digits.
133 * Note extensive error checking of arguments
134 */
135 static long chip8_ioctl(struct file *f, unsigned int cmd,
136                         unsigned long arg)
137 {
138     chip8_opcode op;
139     int isWrite = 0;
140
141     switch (cmd) {
142     case CHIP8_WRITE_ATTR:
143         if (copy_from_user(&op, (chip8_opcode *) arg,
144                           sizeof(chip8_opcode)))
145             return -EACCES;
146         if (!isValidInstruction(op.addr, op.data, 1))
147             return -EINVAL;
148         write_op(op.addr, op.data);
149         break;
150
151     case CHIP8_READ_ATTR:
152         if (copy_from_user(&op, (chip8_opcode *) arg,
153                           sizeof(chip8_opcode)))
154             return -EACCES;

```

```

152     isWrite = isValidInstruction(op.addr, op.data, 0);
153     if(isWrite == 0)
154         return -EINVAL;
155
156
157     if(isWrite == 2)
158         write_op(op.addr, op.data);
159
160     op.readdata = read_value(op.addr);
161     if (copy_to_user((chip8_opcode *) arg, &op,
162                      sizeof(chip8_opcode)))
163         return -EACCES;
164     break;
165
166     default:
167         return -EINVAL;
168     }
169
170     return 0;
171 }
172
173 /* The operations our device knows how to do */
174 static const struct file_operations chip8_fops = {
175     .owner        = THIS_MODULE,
176     .unlocked_ioctl = chip8_ioctl,
177 };
178
179 /* Information about our device for the "misc" framework --
180    like a char dev */
181 static struct miscdevice chip8_misc_device = {
182     .minor        = MISC_DYNAMIC_MINOR,
183     .name         = DRIVER_NAME,
184     .fops         = &chip8_fops,
185 };
186
187 /*
188 * Initialization code: get resources (registers) and
189 * display

```

```

187 * a welcome message
188 */
189 static int __init chip8_probe(struct platform_device *pdev)
190 {
191     int ret;
192
193     /* Register ourselves as a misc device: creates
194      ↳ /dev/chip8 */
195     ret = misc_register(&chip8_misc_device);
196
197     /* Get the address of our registers from the device tree
198      ↳ */
199     ret = of_address_to_resource(pdev->dev.of_node, 0,
200                                 &dev.res);
201     if (ret) {
202         ret = -ENOENT;
203         goto out_deregister;
204     }
205
206     /* Make sure we can use these registers */
207     if (request_mem_region(dev.res.start,
208                           resource_size(&dev.res), DRIVER_NAME) == NULL) {
209         ret = -EBUSY;
210         goto out_deregister;
211     }
212
213     /* Arrange access to our registers */
214     dev.virtbase = of_iomap(pdev->dev.of_node, 0);
215     if (dev.virtbase == NULL) {
216         ret = -ENOMEM;
217         goto out_release_mem_region;
218     }
219
220     /* Write paused state to the chip8 device */
221     write_op(STATE_ADDR, PAUSED_STATE);
222
223     return 0;

```

```

221  out_release_mem_region:
222      release_mem_region(dev.res.start,
223                          ↳ resource_size(&dev.res));
224  out_deregister:
225      misc_deregister(&chip8_misc_device);
226      return ret;
227 }
228 /* Clean-up code: release resources */
229 static int chip8_remove(struct platform_device *pdev)
230 {
231     iounmap(dev.virtbase);
232     release_mem_region(dev.res.start,
233                         ↳ resource_size(&dev.res));
234     misc_deregister(&chip8_misc_device);
235     return 0;
236 }
237 /* Which "compatible" string(s) to search for in the Device
238    Tree */
239 #ifdef CONFIG_OF
240 static const struct of_device_id chip8_of_match[] = {
241     { .compatible = "altr,vga_led" },
242     {} ,
243 };
244 MODULE_DEVICE_TABLE(of, chip8_of_match);
245 #endif
246 /* Information for registering ourselves as a "platform"
247    driver */
248 static struct platform_driver chip8_driver = {
249     .driver      = {
250         .name       = DRIVER_NAME,
251         .owner      = THIS_MODULE,
252         .of_match_table = of_match_ptr(chip8_of_match),
253     },
254     .remove      = __exit_p(chip8_remove),
255 };

```

```

255
256 /* Called when the module is loaded: set things up */
257 static int __init chip8_init(void)
258 {
259     pr_info(DRIVER_NAME ": init\n");
260     return platform_driver_probe(&chip8_driver, chip8_probe);
261 }
262
263 /* Called when the module is unloaded: release resources */
264 static void __exit chip8_exit(void)
265 {
266     platform_driver_unregister(&chip8_driver);
267     pr_info(DRIVER_NAME ": exit\n");
268 }
269
270 module_init(chip8_init);
271 module_exit(chip8_exit);
272
273 MODULE_LICENSE("GPL");
274 MODULE_AUTHOR("The Chip8 Team");
275 MODULE_DESCRIPTION("Chip8 Emulator");

```

### 7.3.3 chip8driver.h

```

1 #ifndef __CHIP8_DRIVER_H__
2 #define __CHIP8_DRIVER_H__
3
4 #include <linux/ioctl.h>
5 #include <stdbool.h>
6
7 typedef struct {
8     unsigned int data;
9     unsigned int addr;
10    unsigned int readdata;
11 } chip8_opcode;
12
13 #define CHIP8_MAGIC 'q'

```

```

14
15 /* ioctls and their arguments */
16 #define CHIP8_WRITE_ATTR _IOW(CHIP8_MAGIC, 1, chip8_opcode
17     ↵    *)
17 #define CHIP8_READ_ATTR  _IOWR(CHIP8_MAGIC, 2, chip8_opcode
18     ↵    *)
19
20 /*
21 * To write data to a particular register, use iowrite with
22 * NNNNNNXX
23 * Where NNNNNN is ignored
24 * XX is the 8 bits to be written
25 *
26 * To read from a register use ioread with one of the
27 *      ↵ following addresses
28 */
29 #define V0_ADDR 0x00
30 #define V1_ADDR 0x04
31 #define V2_ADDR 0x08
32 #define V3_ADDR 0x0C
33 #define V4_ADDR 0x10
34 #define V5_ADDR 0x14
35 #define V6_ADDR 0x18
36 #define V7_ADDR 0x1C
37 #define V8_ADDR 0x20
38 #define V9_ADDR 0x24
39 #define VA_ADDR 0x28
40 #define VB_ADDR 0x2C
41 #define VC_ADDR 0x30
42 #define VD_ADDR 0x34
43 #define VE_ADDR 0x38
44 #define VF_ADDR 0x3C
45
46 /*
47 * To write to the I index register
48 * NNNNDDDD
49 * DDDD is the 16 bits to write
50 */

```

```

49 * Use ioread to read from the I register
50 */
51 #define I_ADDR 0x40

52 /*
53 * To write to the sound timer
54 * NNNNNNDD
55 * Where DD is the number to write to the sound timer
56 *
57 * Use ioread to read from the sound timer
58 */
59 #define SOUND_TIMER_ADDR 0x44

60 /*
61 * To write to the delay timer
62 * NNNNNNDD
63 * Where DD is the number to write to the delay timer
64 *
65 * Use ioread to read from the delay timer
66 */
67 #define DELAY_TIMER_ADDR 0x48

68 /*
69 * To write to the stack pointer
70 * NNNNNNDD
71 * Where DD is the number to write to the stack pointer
72 * Only the last six bits are considered
73 *
74 * Use ioread to read from the stack pointer
75 */
76 #define STACK_POINTER_ADDR 0x60

77 /*
78 * To reset the stack, iowrite
79 */
80 #define STACK_ADDR 0x4C

81 /*
82 * To reset the stack, iowrite
83 */
84 #define STACK_ADDR 0x4C

85 /*
86 */

```

```

87  * To write to the program counter
88  * 0014DDDD
89  * Where DDDD is the number to write to the program counter
90  *
91  * Use ioread to read from the program counter
92  */
93 #define PROGRAM_COUNTER_ADDR 0x50

94
95 /*
96 * To write a keypress to the Chip8 control unit
97 * NNNNNNNP
98 * Where D is the number corresponding to a keypress 0-F
99 * Where P is whether a key is currently pressed or not (0x1,
100    ↳ 0x0)
101 */
101 #define KEY_PRESS_ADDR 0x54

102
103 /*
104 * To change the state of the Chip8
105 * 001600DD
106 * Where DD is an 8-bit number corresponding to varying
107    ↳ states
108 * * 0x01 - Running
109 * * 0x02 - Loading ROM
110 * * 0x03 - Loading font set
111 * * 0x04 - Paused
112 * The state is initially set to loading font set
113 *
114 * Use ioread to read the state of the Chip8
114 */
115 #define STATE_ADDR 0x58
116 #define RUNNING_STATE 0x0
117 #define RUN_INSTRUCTION_STATE 0x1
118 #define PAUSED_STATE 0x2

119
120 /*
121 * To write to a location in memory
122 * 0000_0000_0001_AAAA_AAAA_AAAA_DDDD_DDDD

```

```

123 * Where DD is the 8-bit data that is to be written
124 * Where AAA is the 12-bit address to write the data
125 * Where W is a 1-bit value corresponding to a read or a
   ↳ write
126 *
127 * To read data from memory, use iowrite with
128 * 0000_0000_0000_AAAA_AAAA_AAAA_NNNN_NNNN
129 * Where AAA is the 12-bit address to read the data from
130 */
131 #define MEMORY_ADDR 0x64

132 /*
133 * In order to write data to the framebuffer
134 * 0000_0000_0000_0001_DXXX_XXYY_YYYY
135 * Where XX is the x position (6 bits)
136 * Where YY is the y position (5 bits)
137 * Where DD is the 8-bits of data to write to the screen
138 *
139 * In order to read data from the framebuffer
140 * 0000_0000_0000_0000_NXXX_XXYY_YYYY
141 * Where XX is the x position (6 bits)
142 * Where YY is the y position (5 bits)
143 * Where NN is ignored
144 */
145 #define FRAMEBUFFER_ADDR 0x5C

146 #define SCREEN_HEIGHT 480
147 #define SCREEN_WIDTH 640

148 #define MAX_FBX 32 //32/8
149 #define MAX_FBY 64

150 /*
151 * In order to write data to the instruction
152 * 0000_0000_0000_IIII_IIII_IIII_IIII
153 * Where I corresponds to the 16 bits in the instruction
154 * The state must currently be in Chip8_RUN_INSTRUCTION
155 */

```

```

160 * In order to read data from the framebuffer
161 * 0000_0000_0000_0000_NXXX_XXYY_YYYY
162 * Where XX is the x position (6 bits)
163 * Where YY is the y position (5 bits)
164 * Where NN is ignored
165 */
166 #define INSTRUCTION_ADDR 0x68
167
168 /**
169 * iowrite RESET_ADDR to reset all internal values
170 */
171 #define RESET_ADDR 0x6C
172
173 #endif //__CHIP8_DRIVER_H__

```

### 7.3.4 Makefile

```

1 ifneq (${KERNELRELEASE},)
2 # KERNELRELEASE defined: we are being compiled as part of
3   ↪ the Kernel
4     obj-m := chip8driver.o
5 else
6 # We are being compiled as a module: use the Kernel build
7   ↪ system
8 KERNEL_SOURCE := /usr/src/linux*
9 PWD := $(shell pwd)
10
11 CFLAGS = -Wall
12 OBJECTS = chip8.o usbkeyboard.o
13
14 default: module chip8
15
16 module:
17   ${MAKE} -C ${KERNEL_SOURCE} SUBDIRS=${PWD} modules
18

```

```

19 chip8 : $(OBJECTS)
20     cc $(CFLAGS) -o chip8 $(OBJECTS) -lusb-1.0 -pthread
21
22 lab2.o : lab2.c fbputchar.h usbkeyboard.h
23 usbkeyboard.o : usbkeyboard.c usbkeyboard.h
24
25 .PHONY : clean
26 clean:
27     ${MAKE} -C ${KERNEL_SOURCE} SUBDIRS=${PWD} clean
28     ${RM} chip8
29
30 socfpga.dtb : socfpga.dtb
31     dtc -O dtb -o socfpga.dtb socfpga.dts
32
33 endif

```

### 7.3.5 socfpga.dts

```

1 /*
2  * Copyright (C) 2012 Altera Corporation <www.altera.com>
3  *
4  * This program is free software; you can redistribute it
5  * and/or modify
6  * it under the terms of the GNU General Public License as
7  * published by
8  * the Free Software Foundation; either version 2 of the
9  * License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be
13 * useful,
14 * but WITHOUT ANY WARRANTY; without even the implied
15 * warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
17 * the
18 * GNU General Public License for more details.
19 */

```

```

14 * You should have received a copy of the GNU General Public
15 ← License
16 * along with this program. If not, see
17 ← <http://www.gnu.org/licenses/>.
18 *
19
20 /dts-v1/;
21 /include/ "socfpga.dtsi"
22
23 / {
24     model = "Altera SOC FPGA Cyclone V";
25     compatible = "altr,socfpga-cyclone5", "altr,socfpga";
26
27     chosen {
28         bootargs = "console=ttyS0,57600";
29     };
30
31     memory {
32         name = "memory";
33         device_type = "memory";
34         reg = <0x0 0x40000000>; /* 1 GB */
35     };
36
37     aliases {
38         /* this allow the ethaddr uboot environment variable
39         ← contents
40         * to be added to the gmac1 device tree blob.
41         */
42         ethernet0 = &gmac1;
43     };
44
45     soc {
46         clkmgr@ffd04000 {
47             clocks {
48                 osc1 {

```

```

48                 clock-frequency = <25000000>;
49             };
50         };
51     };
52
53     dcan0: d_can@ffc00000 {
54         status = "disabled";
55     };
56
57     dcan1: d_can@ffc10000 {
58         status = "disabled";
59     };
60
61     dwmmc0@ff704000 {
62         num-slots = <1>;
63         supports-highspeed;
64         broken-cd;
65         altr,dw-mshc-ciу-div = <4>;
66         altr,dw-mshc-sdr-timing = <0 3>;
67
68         slot@0 {
69             reg = <0>;
70             bus-width = <4>;
71         };
72     };
73
74     ethernet@ff700000 {
75         status = "disabled";
76     };
77
78     ethernet@ff702000 {
79         phy-mode = "rgmii";
80         phy-addr = <0xffffffff>; /* probe for phy addr */
81     };
82
83     i2c1: i2c@ffc05000 {
84         status = "disabled";

```

```

85    };
86
87    i2c2: i2c@ffc06000 {
88        status = "disabled";
89    };
90
91    i2c3: i2c@ffc07000 {
92        status = "disabled";
93    };
94
95    qspi: spi@ff705000 {
96        compatible = "cadence,qspi";
97        #address-cells = <1>;
98        #size-cells = <0>;
99        reg = <0xff705000 0x1000>,
100           <0ffa0000 0x1000>;
101        interrupts = <0 151 4>;
102        master-ref-clk = <400000000>;
103        ext-decoder = <0>; /* external decoder */
104        num-chipselect = <4>;
105        fifo-depth = <128>;
106        bus-num = <2>;
107
108        flash0: n25q00@0 {
109            #address-cells = <1>;
110            #size-cells = <1>;
111            compatible = "n25q00";
112            reg = <0>; /* chip select */
113            spi-max-frequency = <100000000>;
114            page-size = <256>;
115            block-size = <16>; /* 2^16, 64KB */
116            quad = <1>; /* 1-support quad */
117            tshsl-ns = <200>;
118            tsd2d-ns = <255>;
119            tchsh-ns = <20>;
120            tslch-ns = <20>;
121

```

```

122         partition@0 {
123             /* 8MB for raw data. */
124             label = "Flash 0 Raw Data";
125             reg = <0x0 0x800000>;
126         };
127         partition@800000 {
128             /* 8MB for jffs2 data. */
129             label = "Flash 0 jffs2 Filesystem";
130             reg = <0x800000 0x800000>;
131         };
132     };
133
134 };
135
136 sysmgr@ffd08000 {
137     cpu1-start-addr = <0xffd080c4>;
138 };
139
140 timer0@ffc08000 {
141     clock-frequency = <100000000>;
142 };
143
144 timer1@ffc09000 {
145     clock-frequency = <100000000>;
146 };
147
148 timer2@ffd00000 {
149     clock-frequency = <25000000>;
150 };
151
152 timer3@ffd01000 {
153     clock-frequency = <25000000>;
154 };
155
156 serial0@ffc02000 {
157     clock-frequency = <100000000>;
158 };

```

```

159
160     serial1@ffc03000 {
161         clock-frequency = <100000000>;
162     };
163
164     usb0: usb@ffb00000 {
165         status = "disabled";
166     };
167
168     usb1: usb@ffb40000 {
169         ulpi-ddr = <0>;
170     };
171
172     i2c0: i2c@ffc04000 {
173         speed-mode = <0>;
174     };
175
176     leds {
177         compatible = "gpio-leds";
178         hps0 {
179             label = "hps_led0";
180             gpios = <&gpio1 15 1>;
181         };
182
183         hps1 {
184             label = "hps_led1";
185             gpios = <&gpio1 14 1>;
186         };
187
188         hps2 {
189             label = "hps_led2";
190             gpios = <&gpio1 13 1>;
191         };
192
193         hps3 {
194             label = "hps_led3";
195             gpios = <&gpio1 12 1>;

```

```

196         };
197     };
198
199     lightweight_bridge: bridge@0xff200000 {
200         #address-cells = <1>;
201         #size-cells = <1>;
202         ranges = < 0x0 0xff200000 0x200000 >;
203
204         compatible = "simple-bus";
205
206         chip8: chip8@0 {
207             compatible = "altr,chip8";
208             reg = <0x0 0x2>;
209         };
210     };
211 };
212
213
214 &i2c0 {
215     lcd: lcd@28 {
216         compatible = "newhaven,nhd-0216k3z-nsw-bbw";
217         reg = <0x28>;
218         height = <2>;
219         width = <16>;
220         brightness = <8>;
221     };
222
223     eeprom@51 {
224         compatible = "atmel,24c32";
225         reg = <0x51>;
226         pagesize = <32>;
227     };
228
229     rtc@68 {
230         compatible = "dallas,ds1339";
231         reg = <0x68>;
232     };

```

233 | };

### 7.3.6 usbkeyboard.c

```
1 #include "usbkeyboard.h"
2
3 #include <stdio.h>
4 #include <stdlib.h>
5
6 /* References on libusb 1.0 and the USB HID/keyboard
7    protocol
8   *
9   * http://libusb.org
10  *
11  * http://www.dreamincode.net/forums/topic/148707-introduction-to-using-libusb
12  * http://www.usb.org/developers/devclass_docs/HID1_11.pdf
13  * http://www.usb.org/developers/devclass_docs/Hut1_11.pdf
14  */
15
16 /**
17  * Find and return a USB keyboard device or NULL if not
18  * found
19  * The argument con
20  *
21  */
22
23 struct libusb_device_handle *openkeyboard(uint8_t
24     *endpoint_address) {
25     libusb_device **devs;
26     struct libusb_device_handle *keyboard = NULL;
27     struct libusb_device_descriptor desc;
28     ssize_t num_devs, d;
29     uint8_t i, k;
30
31     /* Start the library */
32     if ( libusb_init(NULL) < 0 ) {
33         fprintf(stderr, "Error: libusb_init failed\n");
34         exit(1);
35 }
```

```

30 }
31
32 /* Enumerate all the attached USB devices */
33 if ( (num_devs = libusb_get_device_list(NULL, &devs)) < 0
34   ) {
35   fprintf(stderr, "Error: libusb_get_device_list
36   failed\n");
37   exit(1);
38 }
39
40 /* Look at each device, remembering the first HID device
41   that speaks
42   the keyboard protocol */
43
44 for (d = 0 ; d < num_devs ; d++) {
45   libusb_device *dev = devs[d];
46   if ( libusb_get_device_descriptor(dev, &desc) < 0 ) {
47     fprintf(stderr, "Error:
48     libusb_get_device_descriptor failed\n");
49     exit(1);
50   }
51
52   if (desc.bDeviceClass == LIBUSB_CLASS_PER_INTERFACE)
53   {
54     struct libusb_config_descriptor *config;
55     libusb_get_config_descriptor(dev, 0, &config);
56     for (i = 0 ; i < config->bNumInterfaces ; i++)
57       for ( k = 0 ; k <
58         config->interface[i].num_altsetting ; k++)
59       {
60         const struct libusb_interface_descriptor
61         *inter =
62         config->interface[i].altsetting + k ;
63         if ( inter->bInterfaceClass ==
64             LIBUSB_CLASS_HID &&
65             inter->bInterfaceProtocol ==
66             USB_HID_KEYBOARD_PROTOCOL) {
67           int r;

```

```

58         if ((r = libusb_open(dev, &keyboard)) !=
59             0) {
60             fprintf(stderr, "Error: libusb_open
61                   failed: %d\n", r);
62             exit(1);
63         }
64         libusb_set_auto_detach_kernel_driver(
65             i);
66         if ((r = libusb_claim_interface(keyboard,
67             i)) != 0) {
68             fprintf(stderr, "Error:
69                 libusb_claim_interface failed:
70                 %d\n", r);
71             exit(1);
72         }
73     }
74 }
75
76 found:
77 libusb_free_device_list(devs, 1);
78
79 return keyboard;
80 }
81
82 /*
83 * Check to see if any value in the keypad is currently
84 * pressed
85 */

```

```

85 int kbiskeypad(struct usb_keyboard_packet* packet, char
86   ↳ val[1]) {
87     uint8_t keycode = packet->keycode[0];
88
89     switch(keycode) {
90       case KEY1: val[0] = 0x1; return 1; break;
91       case KEY2: val[0] = 0x2; return 1; break;
92       case KEY3: val[0] = 0x3; return 1; break;
93       case KEYC: val[0] = 0xC; return 1; break;
94       case KEY4: val[0] = 0x4; return 1; break;
95       case KEY5: val[0] = 0x5; return 1; break;
96       case KEY6: val[0] = 0x6; return 1; break;
97       case KEYD: val[0] = 0xD; return 1; break;
98       case KEY7: val[0] = 0x7; return 1; break;
99       case KEY8: val[0] = 0x8; return 1; break;
100      case KEY9: val[0] = 0x9; return 1; break;
101      case KEYE: val[0] = 0xE; return 1; break;
102      case KEYA: val[0] = 0xA; return 1; break;
103      case KEY0: val[0] = 0x0; return 1; break;
104      case KEYB: val[0] = 0xB; return 1; break;
105      case KEYF: val[0] = 0xF; return 1; break;
106      default: break;
107    }
108
109    return 0;
110  }
111
112  int kbisstart(struct usb_keyboard_packet* packet) {
113    uint8_t keycode = packet->keycode[0];
114    return keycode == KEY_START;
115  }
116
117  int kbispause(struct usb_keyboard_packet* packet) {
118    uint8_t keycode = packet->keycode[0];
119    return keycode == KEY_PAUSE;
120  }
121  int kbisreset(struct usb_keyboard_packet* packet) {

```

```

122     uint8_t keycode = packet->keycode[0];
123     return keycode == KEY_RESET;
124 }
```

### 7.3.7 usbkeyboard.h

```

1  #ifndef _USBKEYBOARD_H
2  #define _USBKEYBOARD_H
3
4  #include <libusb-1.0/libusb.h>
5
6  #define USB_HID_KEYBOARD_PROTOCOL 1
7
8  /* Modifier bits */
9  #define USB_LCTRL  (1 << 0)
10 #define USB_LSHIFT (1 << 1)
11 #define USB_LALT  (1 << 2)
12 #define USB_LGUI  (1 << 3)
13 #define USB_RCTRL (1 << 4)
14 #define USB_RSHIFT (1 << 5)
15 #define USB_RALT  (1 << 6)
16 #define USB_RGUI  (1 << 7)
17
18
19 /*
20 * Keyboard layout for the Chip8:
21 *      +-----+
22 *      | 1 2 3 C |
23 *      | 4 5 6 D |
24 *      | 7 8 9 E |
25 *      | A O B F |
26 *      +-----+
27 * In this program mapped to a qwerty keyboard:
28 *      +-----+
29 *      | 1 2 3 4 |
30 *      | Q W E R |
31 *      | A S D F |
```

```

32 *      / Z X C V /
33 *      +-----+
34 * Relying on the ascii mapping defined by the usb standard
35 */
36 #define KEY1 0x1E
37 #define KEY2 0x1F
38 #define KEY3 0x20
39 #define KEYC 0x21
40 #define KEY4 0x14
41 #define KEY5 0x1A
42 #define KEY6 0x08
43 #define KEYD 0x15
44 #define KEY7 0x04
45 #define KEY8 0x16
46 #define KEY9 0x07
47 #define KEYE 0x09
48 #define KEYA 0x1d
49 #define KEYO 0x1b
50 #define KEYB 0x06
51 #define KEYF 0x19

52 /*
53 * Three additional keys will be defined
54 * START - Enter key
55 * PAUSE - P key
56 * RESET - O key
57 */
58

59
60 #define KEY_START 0x28
61 #define KEY_PAUSE 0x13
62 #define KEY_RESET 0x12

63
64 struct usb_keyboard_packet {
65     uint8_t modifiers;
66     uint8_t reserved;
67     uint8_t keycode[6];
68 };
69

```

```
70  /* Find and open a USB keyboard device. Argument should
71   * point to
72   * space to store an endpoint address. Returns NULL if no
73   * keyboard
74   * device was found. */
75  extern struct libusb_device_handle *openkeyboard(uint8_t *);
76  int kbiskeypad(struct usb_keyboard_packet* packet, char
77   * val[1]);
78  int kbisstart(struct usb_keyboard_packet* packet);
79  int kbispause(struct usb_keyboard_packet* packet);
80  int kbisreset(struct usb_keyboard_packet* packet);
81
82 #endif
```

## 7.4 Git commit history

```
1 commit 9cbd68476bd8045e95e91276e283cbf7532033a8
2 Author: David Watkins <djw2146@columbia.edu>
3 Date:   Wed May 11 03:26:54 2016 -0400
4
5 Working version of Chip8 EMulator
6
7 commit 4cb0402e07bc27139bb2ec94ed30b98be097cab5
8 Author: David Watkins <djw2146@columbia.edu>
9 Date:   Wed May 11 02:34:55 2016 -0400
10
11 Added double buffering
12
13 commit 10e88a207f36cec3a484a047409808596004284c
14 Author: David Watkins <djw2146@columbia.edu>
15 Date:   Wed May 11 00:35:08 2016 -0400
16
17 Added +2 to stack when popping
18
19 commit 98026152ae7b99b62a3af3b9d7f37b16bcb71472
20 Author: lpo1234 <lpo1017@frontiernet.net>
21 Date:   Tue May 10 22:14:00 2016 -0400
22
23 Theoretical fix for infinite keypress waiting.
24
25 commit 680448658b65edfc694ef473da867d2d05be4fb3
26 Author: David Watkins <djw2146@columbia.edu>
27 Date:   Tue May 10 22:00:27 2016 -0400
28
29 Added stuff
30
31 commit b215d0325e08dff2d67519b258eb21cde202e898
32 Author: David Watkins <djw2146@columbia.edu>
33 Date:   Tue May 10 20:44:38 2016 -0400
34
35 Added correct PC_writedata case
36
37 commit 14fa0ec84b11bb1fc3713d59a511428a3d529bf
38 Author: David Watkins <djw2146@columbia.edu>
```

```
39 Date: Tue May 10 20:42:43 2016 -0400
40
41 Fixed enums
42
43 commit 1778fa04cc1ef9eedb8309aca2de91de227fa9c7
44 Merge: 1647c31 1b9a333
45 Author: David Watkins <djw2146@columbia.edu>
46 Date: Tue May 10 20:36:59 2016 -0400
47
48 Merge branch 'master' of
→ https://github.com/DavidWatkins/Chip8-SystemVerilog
49
50 commit 1647c3141e4ed1cff44e8e7af2a0f07106b7bd5f
51 Author: David Watkins <djw2146@columbia.edu>
52 Date: Tue May 10 20:36:37 2016 -0400
53
54 Added new changes to software
55
56 commit 1b9a333ffde12cf77340a763c1c69e3d6635252b
57 Merge: 6be3433 9aafb45
58 Author: lpo1234 <lpo1017@frontiernet.net>
59 Date: Tue May 10 20:33:28 2016 -0400
60
61 Revamped CPU and Top after memory file findings. Pushing
→ representative stack testbench.
62
63 commit 6be34339a828d7f32c80dc684393dbceac509f50
64 Author: lpo1234 <lpo1017@frontiernet.net>
65 Date: Tue May 10 20:27:00 2016 -0400
66
67 Revamped CPU and some Top to fix with memory module. Adding
→ representative stack testbench.
68
69 commit 9aafb4524c6b3e45f6acf553fc08cf8440b4c3
70 Author: David Watkins <djw2146@columbia.edu>
71 Date: Tue May 10 18:51:22 2016 -0400
72
73 Added missing enums
```

```
74
75 commit b71d4db1b9469cad2e763d550fc0f46e0c6f8c8a
76 Author: David Watkins <djw2146@columbia.edu>
77 Date: Tue May 10 15:59:10 2016 -0400
78
79 Added changed stuff
80
81 commit cc8bc08af9a933ca683d01b0b0b2fb45495490fa
82 Merge: e9f7318 e1cd8d1
83 Author: lpo1234 <lpo1017@frontiernet.net>
84 Date: Mon May 9 20:19:39 2016 -0400
85
86 Merge branch 'master' of
     ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
87
88 commit e9f73189e4d6b106bb55e6d1d9112f23d9938ed0
89 Author: lpo1234 <lpo1017@frontiernet.net>
90 Date: Mon May 9 20:19:20 2016 -0400
91
92 Possible draw-sprite fix.
93
94 commit e1cd8d18d6439d85845f1f281d040a3163f5ccbd
95 Author: Ashley Kling <ask2203@columbia.edu>
96 Date: Mon May 9 18:43:23 2016 -0400
97
98 added tests for 1 and 2, expanded a CPU instruction to
     ↳ accomodate stack
99
100 commit 67d9cb4db4bf466ad8e0c16fa2de5d1fcfaae741
101 Merge: d9eed9c 1bff01e
102 Author: Ashley Kling <ask2203@columbia.edu>
103 Date: Mon May 9 18:40:22 2016 -0400
104
105 Merge branch 'master' of
     ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
106
107 commit 1bff01e901e62f68ec67f276c111c94e0b75a832
108 Author: lpo1234 <lpo1017@frontiernet.net>
```

```
109 Date: Mon May 9 18:39:38 2016 -0400
110
111 Updated big CPU testbench
112
113 commit 2b8da46248698d2952be6ae234e2613a03dccf1e
114 Author: lpo1234 <lpo1017@frontiernet.net>
115 Date: Mon May 9 18:00:04 2016 -0400
116
117 Took out multiplier in led emulator. Small fixes for mem
→ writing in Chip8_Top.
118
119 commit d9eed9ce7aae3db22a66af52f19c133b49498317
120 Author: Ashley Kling <ask2203@columbia.edu>
121 Date: Mon May 9 16:31:30 2016 -0400
122
123 updated 00EE in CPU, added that test to testbench.
124
125 commit 839b32f08ebb3730f0e67d8a08aa03fe228b03e9
126 Author: lpo1234 <lpo1017@frontiernet.net>
127 Date: Mon May 9 15:36:28 2016 -0400
128
129 Fixed simple typo. -_-
130
131 commit fc3d05aa7a2c74c29b18bb0b21b83751dc6ea09
132 Author: lpo1234 <lpo1017@frontiernet.net>
133 Date: Sun May 8 18:37:50 2016 -0400
134
135 Bug fixes in CPU-Top memory request interface. Switched instrs
→ Fx65/Fx55 so they're right.
136
137 commit 99deaae0b4539d29b3ceb3eb0f390a14ac34e0
138 Author: lpo1234 <lpo1017@frontiernet.net>
139 Date: Sun May 8 16:58:37 2016 -0400
140
141 Ash found small bug in Ddyn mem-request pattern. Fixed.
142
143 commit d197eca1ba52494f9445ee34d2635c71a0e75b4b
144 Author: David Watkins <djw2146@columbia.edu>
```

```
145 Date: Sun May 8 09:52:38 2016 -0400
146
147 Chip8 now outputs characters
148
149 commit 0058cc0831b800947b6b85f8b80f4f563a5a086a
150 Merge: d8cc820 2e1e52c
151 Author: David Watkins <djw2146@columbia.edu>
152 Date: Sun May 8 05:45:09 2016 -0400
153
154 Merge branch 'master' of
    ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
155
156 commit d8cc820ec98ee24a27b830187f76125895a1223d
157 Author: David Watkins <djw2146@columbia.edu>
158 Date: Sun May 8 05:44:43 2016 -0400
159
160 Changed top level files
161
162 commit 2e1e52cfba28bce7b60bbc84277c0de3e4d46dd8
163 Author: lpo1234 <lpo1017@frontiernet.net>
164 Date: Sun May 8 05:40:44 2016 -0400
165
166 Updated cpu big testbench to reflect Dxyn isDrawing flag.
167
168 commit 7aa045ae83ba86b0d7abdd8765947133d2055667
169 Author: David Watkins <djw2146@columbia.edu>
170 Date: Sun May 8 05:28:17 2016 -0400
171
172 Added isDrawing flag
173
174 commit 9f51f4a9b46295363cf6bf1519678000f014be25
175 Author: lpo1234 <lpo1017@frontiernet.net>
176 Date: Sun May 8 04:59:58 2016 -0400
177
178 Added a few testing instructions.
179
180 commit ac45ce0a7523b032def876b7aa6397256199528d
181 Merge: 77f28cb d94043e
```

```
182 Author: lpo1234 <lpo1017@frontiernet.net>
183 Date: Sun May 8 02:54:36 2016 -0400
184
185 Merge branch 'master' of
186   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
187 commit 77f28cb7988ed39885f5a3a29bfdda65e91636bd
188 Author: lpo1234 <lpo1017@frontiernet.net>
189 Date: Sun May 8 02:52:11 2016 -0400
190
191 Added big testbench for CPU. Tested Dxyn draw-sprite::it works.
192
193 commit d94043ea52b16bf74ec6d5514691f516681d9008
194 Author: Ashley Kling <ask2203@columbia.edu>
195 Date: Sun May 8 02:19:05 2016 -0400
196
197 added a cpu fix and tests for instructions 3-5 and 9-E
198   ↳ (inclusive)
199 commit fa9819b7a998b5ff092a984cde2e2ba2132659d3
200 Author: Ashley Kling <ask2203@columbia.edu>
201 Date: Sun May 8 01:16:38 2016 -0400
202
203 8-series cpu testbench + alu fix
204
205 commit 89203c16085cd8708c700188b375b7d85808d485
206 Author: lpo1234 <lpo1017@frontiernet.net>
207 Date: Sun May 8 01:02:16 2016 -0400
208
209 Cleaned up Dxyn draw sprite cmd. Still untested.
210
211 commit d5049e1a6185356d3af2f74076267fd49ebb99eb
212 Merge: 85cd6a8 22030b6
213 Author: lpo1234 <lpo1017@frontiernet.net>
214 Date: Sun May 8 00:56:13 2016 -0400
215
216 Merge branch 'master' of
217   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
```

```
217
218 commit 85cd6a80b1e954573d7fa240c019834ef1b7b828
219 Author: lpo1234 <lpo1017@frontiernet.net>
220 Date: Sun May 8 00:54:43 2016 -0400
221
222 Changed how draw-sprite-command Ddyn works. Untested.
223
224 commit 22030b63e776204af143883c4f288e194c65b43d
225 Author: David Watkins <djw2146@columbia.edu>
226 Date: Sun May 8 00:09:25 2016 -0400
227
228 Added proper key reading
229
230 commit fe2f4e96b21c6db9acc523832381fa6d227c44c2
231 Merge: 85e2d54 d63d5cb
232 Author: David Watkins <djw2146@columbia.edu>
233 Date: Sat May 7 21:49:22 2016 -0400
234
235 Merge branch 'master' of
    ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
236
237 commit 85e2d54f55d3a089d181bc78796ff2ba45c787f1
238 Author: David Watkins <djw2146@columbia.edu>
239 Date: Sat May 7 21:49:02 2016 -0400
240
241 Added reading and writing proper functionality
242
243 commit d63d5cb3591f7d481d0920abe76425f3113bc6db
244 Author: Ashley Kling <ask2203@columbia.edu>
245 Date: Sat May 7 19:43:33 2016 -0400
246
247 basic framebuffer testbench w/rudimentary reset
248
249 commit e676f44becd5e6dd449422d014e027b4f064f495
250 Author: David Watkins <djw2146@columbia.edu>
251 Date: Sat May 7 00:29:51 2016 -0400
252
253 Added change to ispressed
```

```
254
255 commit 25466202fab00ebc180bc529d0163dea458b7b95
256 Author: lpo1234 <lpo1017@frontiernet.net>
257 Date: Sat May 7 00:27:30 2016 -0400
258
259 Properly addressed enums.svh.
260
261 commit 6e7ee698ebe3ef5588029f59c26d14b95288ba3a
262 Author: lpo1234 <lpo1017@frontiernet.net>
263 Date: Sat May 7 00:15:20 2016 -0400
264
265 Forgot to push CPU.
266
267 commit cc7eb43fac64d27a259030851aa15aee4e854f10
268 Merge: e7e44fb ad62fb7
269 Author: lpo1234 <lpo1017@frontiernet.net>
270 Date: Sat May 7 00:13:11 2016 -0400
271
272 Merge branch 'master' of
    ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
273
274 commit e7e44fb5c1a6e543d3bce3a76a5e4a724b3d13b9
275 Author: lpo1234 <lpo1017@frontiernet.net>
276 Date: Sat May 7 00:12:43 2016 -0400
277
278 Added stack in. Updated top-level to reflect I/O changes.
    ↳ Prepping for compile.
279
280 commit ad62fb76fe236d96018b981edb80b1485dc98e
281 Author: Ashley Kling <ask2203@columbia.edu>
282 Date: Fri May 6 22:43:38 2016 -0400
283
284 moved some declarations so it works
285
286 commit 965d008c4f4b735c9c4c44989aca249a425112bd
287 Author: Ashley Kling <ask2203@columbia.edu>
288 Date: Fri May 6 22:20:59 2016 -0400
289
```

```
290 updated stack stuff
291
292 commit 6ef322275e702af5f30edd9aacee10c5e494838f
293 Author: gabriellet <gat2118@columbia.edu>
294 Date:   Fri May 6 21:00:32 2016 -0400
295
296 stack testbench updated, stack passes tests
297
298 commit 858effa51dc45c746a19a737d797821fdc020183
299 Merge: ec6a883 960f694
300 Author: gabriellet <gat2118@columbia.edu>
301 Date:   Fri May 6 20:04:56 2016 -0400
302
303 Merge branch 'master' of
304     ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
305 commit ec6a8838916b47857443fde5bcd9191ed1a49f27
306 Author: gabriellet <gat2118@columbia.edu>
307 Date:   Fri May 6 20:04:28 2016 -0400
308
309 Stack testbench update
310
311 commit 960f6947b523d1a2c36a6aef02f214852aa27076
312 Author: David Watkins <djw2146@columbia.edu>
313 Date:   Fri May 6 19:39:13 2016 -0400
314
315 Added changes to makefile
316
317 commit 94932c1591f2e6bd5f5560a5600adb5e254749b6
318 Author: David Watkins <djw2146@columbia.edu>
319 Date:   Fri May 6 14:21:14 2016 -0400
320
321 Added changes to chip8_top
322
323 commit cdcfb9ff9fdecfc322e74b0b71f7db80ea6536eb
324 Author: David Watkins <djw2146@columbia.edu>
325 Date:   Fri May 6 14:17:36 2016 -0400
326
```

```
327 Added changes to stack to allow for multiple states
328
329 commit ff1990098338ff401861a4bd34074ebe104ea68f
330 Author: David Watkins <djw2146@columbia.edu>
331 Date:   Fri May 6 13:21:28 2016 -0400
332
333 Added compiling version of the driver
334
335 commit c7dd59eb78938c38f87a938f33492d28417971eb
336 Author: David Watkins <djw2146@columbia.edu>
337 Date:   Fri May 6 03:27:37 2016 -0400
338
339 Added skeleton code for new stack ops
340
341 commit 5a709b0648979746203101f37ad921df79a42c5a
342 Author: David Watkins <djw2146@columbia.edu>
343 Date:   Fri May 6 03:26:41 2016 -0400
344
345 Added Stack operations
346
347 commit 30e93e3845b365ce2722ca3011e4d71d6925a89c
348 Merge: c5845b6 4baf65a
349 Author: David Watkins <djw2146@columbia.edu>
350 Date:   Thu May 5 02:48:03 2016 -0400
351
352 Merge branch 'master' of
353   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
354 commit c5845b6ca4bc6a12674a44a4f5ffb29fe53e560a
355 Author: David Watkins <djw2146@columbia.edu>
356 Date:   Thu May 5 02:47:24 2016 -0400
357
358 Added bash file that can run modelsim from command line
359
360 commit 4baf65abbe6e1da8dbe6479f7c47de365c859bd1
361 Author: lpo2105 <lpo2105@micro6.ilab.columbia.edu>
362 Date:   Thu May 5 01:43:37 2016 -0400
363
```

```
364 Added greater depth to testing approach.  
365  
366 commit 9f67225ddcb4bb9b0aa54204c612d6d1def0cdd8  
367 Author: lpo2105 <lpo2105@micro6.ilab.columbia.edu>  
368 Date: Thu May 5 00:48:24 2016 -0400  
369  
370 Adding CPU testbench for 6xkk/7xkk. Template to be extended to  
→ most CPU cmds.  
371  
372 commit d47c1b3833026dffca2c7fd54dc3783989e3bbc6  
373 Author: David Watkins <djw2146@columbia.edu>  
374 Date: Tue May 3 17:24:08 2016 -0400  
375  
376 Removed "Hello world"  
377  
378 commit 83bd4ce83c55a21172903bd2cbc6ead0f6e4077a  
379 Author: David Watkins <djw2146@columbia.edu>  
380 Date: Tue May 3 17:23:08 2016 -0400  
381  
382 All tests pass  
383  
384 commit 761072efe624954cf5cfa0438fc8b8d2c18a632f  
385 Author: David Watkins <djw2146@columbia.edu>  
386 Date: Tue May 3 16:48:15 2016 -0400  
387  
388 Added testbench update  
389  
390 commit a79a5420cbbaf24d86f84b92bf4a2aaf0627c7  
391 Merge: 35dfb69 53d84f5  
392 Author: David Watkins <djw2146@columbia.edu>  
393 Date: Tue May 3 16:14:01 2016 -0400  
394  
395 Merge branch 'master' of  
→ https://github.com/DavidWatkins/Chip8-SystemVerilog  
396  
397 commit 53d84f54d5b3051834b034ad32f0cf4f930347d7  
398 Merge: 22fe90c ed8e36c  
399 Author: Ashley Kling <ask2203@columbia.edu>
```

```

400 Date: Tue May 3 15:26:36 2016 -0400
401
402 Merge branch 'master' of
403   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
404 commit 22fe90c4e442c5e6dee85b7eda89162f04ff6303
405 Author: Ashley Kling <ask2203@columbia.edu>
406 Date: Tue May 3 15:26:03 2016 -0400
407
408 cleaned up stack testbench
409
410 commit ed8e36c7356e3f099418e2888de466e1d0eac4d9
411 Merge: 719a978 cf347a1
412 Author: gabriellet <gat2118@columbia.edu>
413 Date: Tue May 3 15:14:54 2016 -0400
414
415 Merge branch 'master' of
416   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
417 commit 719a978d4fc00790ce49149ca0e4f075afddfc4
418 Author: gabriellet <gat2118@columbia.edu>
419 Date: Tue May 3 15:14:28 2016 -0400
420
421 fb testbench update
422
423 commit cf347a195fa9023bc25c12e8f7a241f1549d48cb
424 Author: Ashley Kling <ask2203@columbia.edu>
425 Date: Tue May 3 15:05:38 2016 -0400
426
427 Stack working. When reading, output will be available on second
428   ↳ clock cycle. When enable = 00, it will read the value at
429   ↳ the 0th place in the stack, ignore this.
430
431 commit cb46529016f72bf7410f10404df9fedf762b614f
432 Author: gabriellet <gat2118@columbia.edu>
433 Date: Tue May 3 13:39:19 2016 -0400
434
435 tested ALU update

```

```

434
435 commit 87816742da290e4f79ade8934a370eed163c8f3d
436 Merge: 5a914f3 36e18d2
437 Author: gabriellet <gat2118@columbia.edu>
438 Date: Tue May 3 13:37:37 2016 -0400
439
440 Merge branch 'master' of
441   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
442 commit 5a914f300ad9a6a9580cf4ad60c51deba4863ffd
443 Author: gabriellet <gat2118@columbia.edu>
444 Date: Tue May 3 13:37:15 2016 -0400
445
446 updated tested ALU and enum
447
448 commit 35dfb697a5c5c9c4f7c483846fc381ade5bfb5f0
449 Merge: 4165b3d 36e18d2
450 Author: David Watkins <djw2146@columbia.edu>
451 Date: Tue May 3 13:33:03 2016 -0400
452
453 Merge branch 'master' of
454   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
455 commit 36e18d283f432890f757638157050543925e3419
456 Author: Gabrielle A Taylor <gat2118@columbia.edu>
457 Date: Tue May 3 13:26:56 2016 -0400
458
459 Update .gitignore
460
461 commit 1c7e5013d058406d0db3dc80bb155ca6892206ee
462 Author: Gabrielle A Taylor <gat2118@columbia.edu>
463 Date: Tue May 3 13:22:56 2016 -0400
464
465 Fixed carry in ALU_f_ADD
466
467 commit 95b7b78a1768822c00fd4a3ef263b7bafde03535
468 Author: gabriellet <gat2118@columbia.edu>
469 Date: Tue May 3 13:11:33 2016 -0400

```

```
470
471 final ALU files - confirmed working
472
473 commit 8275d351c7b51571213c949e39a756ebf58a140f
474 Author: gabriellet <gat2118@columbia.edu>
475 Date:   Sat Apr 30 23:34:40 2016 -0400
476
477 Stack testbench intermediate files:
478
479 commit 4c40781ff01ebe970ed4869a15cacf2832de7037
480 Author: gabriellet <gat2118@columbia.edu>
481 Date:   Sat Apr 30 23:32:19 2016 -0400
482
483 ALU testbench files updated
484
485 commit 7ed35963899dce144ed4292b45bb16eed78d9ce0
486 Author: gabriellet <gat2118@columbia.edu>
487 Date:   Sat Apr 30 17:21:20 2016 -0400
488
489 additional ALU test files
490
491 commit c6faa7efd3dc2d29ffaea21853acd405822b76f7
492 Author: gabriellet <gat2118@columbia.edu>
493 Date:   Sat Apr 30 17:19:56 2016 -0400
494
495 ALU testing underway, still buggy
496
497 commit 2eab7a62d7b455e38b324c723b461ec29e0c6d53
498 Author: gabriellet <gat2118@columbia.edu>
499 Date:   Thu Apr 28 16:06:36 2016 -0400
500
501 new testbench files
502
503 commit e4f8c1ef00d2144481f1f63f74faa7a4c06e643f
504 Author: gabriellet <gat2118@columbia.edu>
505 Date:   Thu Apr 28 15:49:32 2016 -0400
506
507 moved audio files in test to test/audio
```

```

508
509 commit 529cdcfb17e518f53d2c7b88caffa99ce6b3b1e6
510 Author: gabriellet <gat2118@columbia.edu>
511 Date: Thu Apr 28 15:45:20 2016 -0400
512
513 push stack ram to test/stk
514
515 commit e19c5f02d734e5572249e492e79890ea1f8c978c
516 Author: gabriellet <gat2118@columbia.edu>
517 Date: Thu Apr 28 13:32:43 2016 -0400
518
519 stack testbench update
520
521 commit e5998b9b50fadeecaf85a5abd9697b598b21bb2
522 Merge: 9a46db3 0b17dc7
523 Author: gabriellet <gat2118@columbia.edu>
524 Date: Thu Apr 28 13:14:19 2016 -0400
525
526 Merge branch 'master' of
      ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
527 Stack updated, merging to testbench
528
529 commit 9a46db322947f85a19d149a3fc9fdcff0dbaaa6
530 Author: gabriellet <gat2118@columbia.edu>
531 Date: Thu Apr 28 13:13:56 2016 -0400
532
533 stack_testbench update
534
535 commit 0b17dc7a2c1ce76f303790bab864614746189dba
536 Merge: 3ac5cff 6ddb751
537 Author: Ashley Kling <ask2203@columbia.edu>
538 Date: Thu Apr 28 13:04:17 2016 -0400
539
540 Merge branch 'master' of
      ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
541
542 commit 3ac5cfffa9b0ca446599d195381851d777b5cf4f
543 Author: Ashley Kling <ask2203@columbia.edu>
```

```
544 Date: Thu Apr 28 13:04:04 2016 -0400
545
546 updated stack
547
548 commit 6ddb751b3ba98f4cf5a06c6f36ae5ccd3704771a
549 Author: gabriellet <gat2118@columbia.edu>
550 Date: Thu Apr 28 12:51:14 2016 -0400
551
552 Stack testbench files
553
554 commit 889fbe39b2d2a638312fe15fdb16b5a94c2414ef
555 Author: lpo1234 <lpo1017@frontiernet.net>
556 Date: Thu Apr 28 00:09:17 2016 -0400
557
558 Revamped framebuffer and CPU to match. Theoretically supports
    ↳ draw sprite and clear screen cmds. CHANGES NOT REFLECTED IN
    ↳ Chip8_Top
559
560 commit 18e996d09a296988bfc56018fc85abce34341a2a
561 Author: lpo1234 <lpo1017@frontiernet.net>
562 Date: Wed Apr 27 18:17:44 2016 -0400
563
564 Adding MegaFunction Framebuffer memory.
565
566 commit 2913ad5cfcc62beb48050c2705d375b981103d3d3
567 Author: lpo1234 <lpo1017@frontiernet.net>
568 Date: Tue Apr 26 21:55:28 2016 -0400
569
570 Fixed CPU casex-ladder syntax problems for DC conditions.
    ↳ Cleaned up truncation warnings. Tested
    ↳ triple-ported-register-file. It seems to work.
571
572 commit 4165b3d491665156e77feb45618226961df4cdff
573 Author: David Watkins <djw2146@columbia.edu>
574 Date: Tue Apr 26 14:52:57 2016 -0400
575
576 Added a python script for generating properly formatted MIF
    ↳ files
```

```
577
578 commit 5e6485faef80c816c1ad30c4b84635552b2e4b04
579 Author: David Watkins <djw2146@columbia.edu>
580 Date: Tue Apr 26 08:07:17 2016 -0400
581
582 Updated code to support instructions at top level
583
584 commit dfd8770b0b2b77dca4b54fcfedbc0f806b6e0ac87
585 Author: David Watkins <djw2146@columbia.edu>
586 Date: Tue Apr 26 06:57:58 2016 -0400
587
588 Compiling version of entire project
589
590 commit 932855dd20090b7ad68be05b9c9009ae2d762ec2
591 Author: David Watkins <djw2146@columbia.edu>
592 Date: Tue Apr 26 04:14:34 2016 -0400
593
594 Added updates to the sound controller
595
596 commit aa6f9247e0d2766b790dfafb3c8520e32ca3081b
597 Author: David Watkins <djw2146@columbia.edu>
598 Date: Tue Apr 26 03:46:59 2016 -0400
599
600 Added header files
601
602 commit 7f698e00b0b09c2348541596c9b71568eb5bf98e
603 Author: David Watkins <djw2146@columbia.edu>
604 Date: Tue Apr 26 03:38:25 2016 -0400
605
606 Fixed ALU and CPU to support all instruction (Draw NYI)
607
608 commit 2a1286c317fd71e37bfe17bb83903c56bb340d27
609 Author: lpo1234 <lpo1017@frontiernet.net>
610 Date: Mon Apr 25 01:34:59 2016 -0400
611
612 Added most PC functionality to the CPU.
613
614 commit 0ddddfb9e4ef23b12aa804610d08ce9819a9ddde
```

```
615 Author: lpo1234 <lpo1017@frontiernet.net>
616 Date: Sun Apr 24 23:55:18 2016 -0400
617
618 Added BCD functionality to CPU. Fixed oversized default value
   ↳ in random num generator.
619
620 commit b58aa9ef625b711d4180fedc31c5ce193eccb036
621 Merge: 9cbc6d7 3de4899
622 Author: lpo1234 <lpo1017@frontiernet.net>
623 Date: Sun Apr 24 23:40:36 2016 -0400
624
625 Merge branch 'master' of
   ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
626
627 commit 9cbc6d726f251f8ef79a779751bd8a60dc902c99
628 Author: lpo1234 <lpo1017@frontiernet.net>
629 Date: Sun Apr 24 23:39:50 2016 -0400
630
631 Updated CPU for top-level control. It does not deal with instrs
   ↳ regarding PC, framebuffer, or BCD.
632
633 commit 3de48994fb16a1fa8d2e4952da22f1527fe5153b
634 Author: ask2203 <ask2203@micro15.ilab.columbia.edu>
635 Date: Sat Apr 23 18:13:12 2016 -0400
636
637 untested stack file
638
639 commit b172666ea7b2de978e5c5096882a788f4a0ea9a6
640 Author: Gabrielle A Taylor <gat2118@columbia.edu>
641 Date: Fri Apr 22 18:22:09 2016 -0400
642
643 Duplicated files deleted
644
645 commit 2d0c61a7e144c8759127d99af262cbd7c1140d64
646 Author: Gabrielle A Taylor <gat2118@columbia.edu>
647 Date: Fri Apr 22 18:21:28 2016 -0400
648
649 Duplicated files deleted
```

```
650
651 commit 34e241302a4ef0977bbdbe2e5394f0dc8f6556b6
652 Author: Gabrielle A Taylor <gat2118@columbia.edu>
653 Date:   Fri Apr 22 18:21:08 2016 -0400
654
655 Duplicated files deleted
656
657 commit f0a1dc7c772ed001e2073d755dd7f1d67ce37547
658 Author: Gabrielle A Taylor <gat2118@columbia.edu>
659 Date:   Fri Apr 22 18:20:58 2016 -0400
660
661 Duplicated files deleted
662
663 commit d56f6ab9c26e0d9fd880f727d6be6fbbec44be6e
664 Author: Gabrielle A Taylor <gat2118@columbia.edu>
665 Date:   Fri Apr 22 18:20:47 2016 -0400
666
667 Duplicated files deleted
668
669 commit 92890f725928acce4d7755c0784d69cca85062ac
670 Author: Gabrielle A Taylor <gat2118@columbia.edu>
671 Date:   Fri Apr 22 18:20:34 2016 -0400
672
673 Duplicated files deleted
674
675 commit be03d04f33b42f09ee3dfb45c2b5cd0cf547df4e
676 Author: Gabrielle A Taylor <gat2118@columbia.edu>
677 Date:   Fri Apr 22 18:17:14 2016 -0400
678
679 Script + samples + original wav
680
681 Script extracts samples from wav file. Samples for beep2.wav
   ↳ are in samples.txt
682
683 commit b705539fb6021353ac2ddb476142288cf3f267a1
684 Author: David Watkins <djw2146@columbia.edu>
685 Date:   Tue Apr 19 15:51:50 2016 -0400
686
```

```
687 Added files that were missing from before
688
689 commit 512823e2a46d09a3d8305699f60615b0a0988908
690 Author: David Watkins <djw2146@columbia.edu>
691 Date:   Sat Apr 16 18:11:21 2016 -0400
692
693 Moved testbench for delay timer
694
695 commit 844880913d18dee6e4c4cee38dd8a12c012a413f
696 Author: David Watkins <djw2146@columbia.edu>
697 Date:   Sat Apr 16 18:08:35 2016 -0400
698
699 Redid file directory
700
701 commit f55eb911738c27765f9c0a8f8158ba6393cd541b
702 Author: gabriellet <gat2118@columbia.edu>
703 Date:   Sat Apr 16 07:24:54 2016 -0400
704
705 removed mips directory from test, added delay timer and
    ↳ testbench to test
706
707 commit ef7706c9c01f65935ec1f8d8ba75900d9821b824
708 Merge: 6e306a2 fcb610e
709 Author: gabriellet <gat2118@columbia.edu>
710 Date:   Fri Apr 15 12:22:42 2016 -0400
711
712 Merge branch 'master' of
    ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
713
714 commit 6e306a29a061cb4612c4e1c4c6160b731e7ec84a
715 Author: gabriellet <gat2118@columbia.edu>
716 Date:   Fri Apr 15 12:21:58 2016 -0400
717
718 delay timer implementation
719
720 commit fcb610e9bfb6face733099142e3a1d275106f871
721 Author: lpo1234 <lpo1017@frontiernet.net>
722 Date:   Wed Apr 13 23:50:14 2016 -0400
```

723  
724 Here is the test waveform to show on Thursday Apr 14. I am the  
→ prettiest.  
725  
726 commit fa449ed041a88e713c1f133beb3033e2a5f48021  
727 Author: David Watkins <djw2146@columbia.edu>  
728 Date: Wed Apr 13 23:02:56 2016 -0400  
729  
730 Forgot test  
731  
732 commit 050daf1090a867f8181af4f1db6bbaad9c6054f6  
733 Merge: c37a15a ac284ac  
734 Author: David Watkins <djw2146@columbia.edu>  
735 Date: Wed Apr 13 22:46:39 2016 -0400  
736  
737 Merge branch 'master' of  
→ <https://github.com/DavidWatkins/Chip8-SystemVerilog>  
738  
739 commit c37a15a2e2f3ffd0e0ccdb502336571f20f50000  
740 Author: David Watkins <djw2146@columbia.edu>  
741 Date: Wed Apr 13 22:46:23 2016 -0400  
742  
743 Cries  
744  
745 commit ac284ac58a27087d4766eb4ccd0061ee9e555917  
746 Author: lpo1234 <lpo1017@frontiernet.net>  
747 Date: Wed Apr 13 22:40:15 2016 -0400  
748  
749 Added dual ported memory for register file and memory.  
750  
751 commit 07694cbe2ac21b44c806dc56c466470b44dfdc02  
752 Author: David Watkins <djw2146@columbia.edu>  
753 Date: Wed Apr 13 22:30:14 2016 -0400  
754  
755 Added testbench  
756  
757 commit ade90c0e38478bbeb1ccd93f255fce8834adf321  
758 Author: David Watkins <djw2146@columbia.edu>

```
759 Date: Wed Apr 13 22:03:12 2016 -0400
760
761 Added test rom initialization file that tests basic
    ↳ instructions
762
763 commit 0e31d27cf7e437b01e5e055bb7d90e229c0de5db
764 Author: David Watkins <djw2146@columbia.edu>
765 Date: Wed Apr 13 21:56:58 2016 -0400
766
767 Added a new top level module that runs over the pc
768
769 commit 2133d107428c3ad7ad74065bca527d4e7fb1f2a4
770 Merge: 77d791d 1443ecb
771 Author: David Watkins <djw2146@columbia.edu>
772 Date: Wed Apr 13 21:31:27 2016 -0400
773
774 Merge branch 'master' of
    ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
775
776 commit 77d791dc6a4f63a5a0ea4fecd13c111f1d2d1700
777 Author: David Watkins <djw2146@columbia.edu>
778 Date: Wed Apr 13 21:31:10 2016 -0400
779
780 Added in progress version of chip8 top level
781
782 commit 1443ecb60c1d37b41f097d38155e532bc3c2d5c0
783 Author: Ashley Kling <ask2203@columbia.edu>
784 Date: Wed Apr 13 20:42:55 2016 -0400
785
786 fixed dec-hex
787
788 commit 4ffe66bcffed6e9a54b3c44b7111ea0428327a8f
789 Author: David Watkins <DavidWatkins@users.noreply.github.com>
790 Date: Wed Apr 13 19:02:48 2016 -0400
791
792 Delete SoCKit_Top.ipinfo
793
794 commit dbbdc521e65452464f9d20f66fe9b73828dac53a
```

```
795 Author: David Watkins <DavidWatkins@users.noreply.github.com>
796 Date:   Wed Apr 13 19:02:43 2016 -0400
797
798 Delete SoCKit_Top.sld_design_entry.sci
799
800 commit 4c5ce1ca7dd6541308ef7f923505f80bb075035f
801 Author: David Watkins <DavidWatkins@users.noreply.github.com>
802 Date:   Wed Apr 13 19:02:38 2016 -0400
803
804 Delete SoCKit_Top.db_info
805
806 commit 966343eeb5429e8624554017bb7ac7ba6e62a27c
807 Author: gat2118 <gat2118@micro18.ilab.columbia.edu>
808 Date:   Wed Apr 13 16:37:06 2016 -0400
809
810 copied files to test in order to implement MIPS-like processor
     ↳ design
811
812 commit 6c913287ef3c20a539423fbe0cd6715d7f0a2923
813 Merge: 8048a4a a67a547
814 Author: Ashley Kling <ask2203@columbia.edu>
815 Date:   Tue Apr 12 14:44:31 2016 -0400
816
817 Merge branch 'master' of
     ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
818
819 commit a67a547f37b38d65657e96f8174ba7acf63dfdef
820 Merge: af3718e 9761ee9
821 Author: David Watkins <djw2146@columbia.edu>
822 Date:   Tue Apr 12 14:16:55 2016 -0400
823
824 Merge branch 'master' of
     ↳ https://github.com/DavidWatkins/Chip8-SystemVerilog
825
826 commit af3718ee4f5a685290f46ec466e8562c2d9711ed
827 Author: David Watkins <djw2146@columbia.edu>
828 Date:   Tue Apr 12 14:16:27 2016 -0400
829
```

```
830 Added register file code
831
832 commit 8048a4a57a22b65b5a41c11d1851bbe225e8145a
833 Author: Ashley Kling <ask2203@columbia.edu>
834 Date: Tue Apr 12 13:57:34 2016 -0400
835
836 updated CPU file, need to test, esp # cycles per instruction
837
838 commit 9761ee95f0496f077c326bee4aba1825aeedd73f
839 Author: David Watkins <DavidWatkins@users.noreply.github.com>
840 Date: Tue Apr 12 13:48:02 2016 -0400
841
842 Update README.md
843
844 commit 5cd366cfed3ea15b0c9961b5ca37ee9bf98cb840
845 Author: David Watkins <DavidWatkins@users.noreply.github.com>
846 Date: Tue Apr 12 12:19:30 2016 -0400
847
848 Update README.md
849
850 commit 553cfe6250d772c230278d92c4ec80d0fc230a93
851 Author: David Watkins <DavidWatkins@users.noreply.github.com>
852 Date: Tue Apr 12 12:19:05 2016 -0400
853
854 Update README.md
855
856 commit 02e3c4b657d98c03c53c3f1d7d5bc501ad851dc4
857 Author: David Watkins <djw2146@columbia.edu>
858 Date: Tue Apr 12 16:56:18 2016 +0100
859
860 dded u-boot script which is necessary for getting the sockit
     ↳ board to boot
861
862 commit dd7d50a9e71b0c4f1623d25148c4d96b4790e306
863 Merge: b15ad94 f9f3232
864 Author: David Watkins <djw2146@columbia.edu>
865 Date: Tue Apr 12 03:58:11 2016 -0400
866
```

```

867 Merge branch 'master' of
     → https://github.com/DavidWatkins/Chip8-SystemVerilog
868
869 commit b15ad94df32fd92cbf140653c3f5b44649d32daa
870 Author: David Watkins <djw2146@columbia.edu>
871 Date:   Tue Apr 12 03:57:31 2016 -0400
872
873 Added communication back and forth. Needs testing
874
875 commit f9f3232503cc2a16efe751c24054502fc8ed20c8
876 Author: Gabrielle A Taylor <gat2118@columbia.edu>
877 Date:   Mon Apr 11 14:29:13 2016 -0400
878
879 Update sockit_top.sv
880
881 commit 5ca73ee5a326b6b758c9046ed2cab6b02ed9aa6c
882 Author: Gabrielle A Taylor <gat2118@columbia.edu>
883 Date:   Mon Apr 11 14:15:56 2016 -0400
884
885 Changed implementation
886
887 Clock is now default low (no edge detection necessary for use
     → in delay, sound timers).
888
889 commit 99ac7117f1e962f780f1ff26d8026510292eec02
890 Author: Gabrielle A Taylor <gat2118@columbia.edu>
891 Date:   Mon Apr 11 12:55:44 2016 -0400
892
893 clk_div.sv works
894
895 Seems to work in simulation. Will try a few more tests with
     → edge cases to determine behavior.
896
897 commit 75aa22faed1b3fabcf1f3439d8df2defcded46e95
898 Author: Gabrielle A Taylor <gat2118@columbia.edu>
899 Date:   Mon Apr 11 12:28:07 2016 -0400
900
901 bcd.sv now works

```

```
902
903 Tests indicate module works.
904
905 commit a744ba2ec0ca570e06cecc2770aedd176d044d2d
906 Author: Gabrielle A Taylor <gat2118@columbia.edu>
907 Date: Sun Apr 10 13:30:36 2016 -0400
908
909 Fix compilation errors
910
911 Simulated in modelsim but still does not work. Loop in line 13
→ only runs once. Will investigate why.
912
913 commit fd6b8e93cce11ec12061d7a328f912fbe85f1cad
914 Author: lpo1234 <lpo1017@frontiernet.net>
915 Date: Sun Apr 10 03:32:07 2016 -0400
916
917 Adding memory module waveform test.
918
919 commit 4b5e6b9b12c7e6706af463a5324b2b65ba1626cb
920 Author: lpo1234 <lpo1017@frontiernet.net>
921 Date: Sun Apr 10 03:18:46 2016 -0400
922
923 Added 4096byte memory module.
924
925 commit 90fd2f67f770b0eb25a8930198fdb8b05b8afa88
926 Author: Gabrielle A Taylor <gat2118@columbia.edu>
927 Date: Sat Apr 9 22:52:47 2016 -0400
928
929 BCD module in systemverilog
930
931 commit 6de8348fd274c49e7564113fcf631f53b4cbfc32
932 Author: Gabrielle A Taylor <gat2118@columbia.edu>
933 Date: Sat Apr 9 22:24:38 2016 -0400
934
935 Information relevant to audio test
936
937 commit df18feae1c60ffe03b53e1bf4117c52852969b2b
938 Author: Gabrielle A Taylor <gat2118@columbia.edu>
```

```
939 Date: Sat Apr 9 22:19:38 2016 -0400
940
941 Rename sockit_top.sv to test/sockit_top.sv
942
943 commit 08b5170a0a56c9c93b32fdb4386313c09af62bc2
944 Author: Gabrielle A Taylor <gat2118@columbia.edu>
945 Date: Sat Apr 9 22:19:18 2016 -0400
946
947 Rename i2c_controller.sv to test/i2c_controller.sv
948
949 commit f8d101603af378f5a6d530a1a924ba3a05a383e9
950 Author: Gabrielle A Taylor <gat2118@columbia.edu>
951 Date: Sat Apr 9 22:19:06 2016 -0400
952
953 Rename i2c_av_config.sv to test/i2c_av_config.sv
954
955 commit 2566be1a0a3a9cc017a8ba3829e0894e12fdb3ff
956 Author: Gabrielle A Taylor <gat2118@columbia.edu>
957 Date: Sat Apr 9 22:18:50 2016 -0400
958
959 Rename delay_timer.sv to test/delay_timer.sv
960
961 commit 7a3c6f0788411ac69e24c6680e1104ed22a31167
962 Author: Gabrielle A Taylor <gat2118@columbia.edu>
963 Date: Sat Apr 9 22:18:37 2016 -0400
964
965 Rename audio_effects.sv to test/audio_effects.sv
966
967 commit e4ebe895dc5205aaf655a51849df03bb5f08268c
968 Author: Gabrielle A Taylor <gat2118@columbia.edu>
969 Date: Sat Apr 9 22:18:17 2016 -0400
970
971 Rename audio_codec.sv to test/audio_codec.sv
972
973 commit 070b16f32c9820b0c0d15aca5b321020e9d181fa
974 Author: Gabrielle A Taylor <gat2118@columbia.edu>
975 Date: Sat Apr 9 22:17:52 2016 -0400
976
```

```
977 Test for SoCKit Sound
978
979 Tests whether SoCKit board will output 440Hz sine wave
980
981 commit 1d325b6e8e083bfcd199f182fa298fa31fde876d
982 Author: Gabrielle A Taylor <gat2118@columbia.edu>
983 Date: Sat Apr 9 22:12:39 2016 -0400
984
985 Clock divider
986
987 Converts from 50MHz clock to 60Hz clock. To be used in delay
→ and sound timers.
988
989 commit 51f1409a2ae329f78aa28986d65e4fae7b306df0
990 Author: lpo1234 <lpo1017@frontiernet.net>
991 Date: Thu Apr 7 13:51:30 2016 -0400
992
993 ACTUALLY added initial CPU vwf test.
994
995 commit 174d5f5228965d9067ab7cd326f82fb7ca7ba5d0
996 Author: lpo1234 <lpo1017@frontiernet.net>
997 Date: Thu Apr 7 13:49:56 2016 -0400
998
999 Comb instr decode works. Cleaned code. Adding CPU vwf test.
1000
1001 commit 260a661a7c625be088986997ee5c417edda4ca01
1002 Author: lpo1234 <lpo1017@frontiernet.net>
1003 Date: Thu Apr 7 13:29:00 2016 -0400
1004
1005 Made instruction decode combinational. I believe it works, but
→ not all instrs implemented were tested.
1006
1007 commit 0da757bd8e26d569a3ee7682fbac164fe97339b8
1008 Author: lpo1234 <lpo1017@frontiernet.net>
1009 Date: Tue Apr 5 23:58:29 2016 -0400
1010
1011 Started CPU. Data is not appearing as expected. See
→ CPU_initial_test.vwf waveform.
```

```
1012  
1013 commit 146d5c3b0dcc7412d0853dc1258e1d5ab239a3a2  
1014 Author: lpo1234 <lpo1017@frontiernet.net>  
1015 Date: Tue Apr  5 21:07:43 2016 -0400  
1016  
1017 Created register module to control V0-VF. Also created a folder  
→ to hold test waveforms.  
1018  
1019 commit 904fba10987ee571bae75211cf3992935e10ec45  
1020 Author: lpo1234 <lpo1017@frontiernet.net>  
1021 Date: Thu Mar 31 02:54:44 2016 -0400  
1022  
1023 Added 16b random number generator.  
1024  
1025 commit e3410e4fb4c763867bdb6e503751ed155e06830c  
1026 Author: lpo1234 <lpo1017@frontiernet.net>  
1027 Date: Thu Mar 31 00:07:47 2016 -0400  
1028  
1029 Added ALU and memory-to-screen VGA emulator  
1030  
1031 commit 9b629ea8e3d0dde730a90f39806208222fde5965  
1032 Author: David Watkins <djw2146@columbia.edu>  
1033 Date: Wed Mar 30 16:12:46 2016 -0400  
1034  
1035 Fixed key press values  
1036  
1037 commit 6520961eb44e7fb8d902c80adb6ce8c38807276c  
1038 Merge: ef35672 ad878f1  
1039 Author: David Watkins <djw2146@columbia.edu>  
1040 Date: Wed Mar 30 04:03:29 2016 -0400  
1041  
1042 Merge branch 'master' of  
→ https://github.com/DavidWatkins/Chip8-SystemVerilog  
1043  
1044 commit ef35672328ff42b0b4fcde7d7391555aaa3574da  
1045 Author: David Watkins <djw2146@columbia.edu>  
1046 Date: Wed Mar 30 04:02:55 2016 -0400  
1047
```

```
1048 New status for stuff. Added a bunch of local stuff
1049
1050 commit ad878f13b861ec9b4084a9eaf8d94309be5f1590
1051 Author: David Watkins <DavidWatkins@users.noreply.github.com>
1052 Date:   Tue Mar 29 23:50:36 2016 -0400
1053
1054 Update Chip8_VGA_Emulator.sv
1055
1056 commit 7b43eb8ec84468304dbae5eba2a739278eb2fc16
1057 Author: David Watkins <djw2146@columbia.edu>
1058 Date:   Tue Mar 29 23:34:08 2016 -0400
1059
1060 Added changes to framebuffer code
1061
1062 commit cab0f99bfc540cd1a64a27a29ba18d5c68818b32
1063 Author: David Watkins <djw2146@columbia.edu>
1064 Date:   Mon Mar 28 23:34:20 2016 -0400
1065
1066 Initial commit
1067
1068 commit 7e436e72239b592fb76c50e6d126497341210cea
1069 Author: David Watkins <DavidWatkins@users.noreply.github.com>
1070 Date:   Mon Mar 28 17:23:05 2016 -0400
1071
1072 Initial commit
```

## 7.5 Schematics

### 7.5.1 Chip8\_framebuffer.sv



### 7.5.2 clk\_div.sv



### 7.5.3 timer.sv



#### 7.5.4 Chip8\_rand\_num\_generator.sv



### 7.5.5 bcd.sv



### 7.5.6 memory.sv



### 7.5.7 Chip8\_SoundController.sv



### 7.5.8 Entire Design



















































































