# A RISC-V computer system design

Latest version available at: http://www.yourhomepage.com

Xiong Juncheng

December 12, 2023

## Abstract

In this project, I designed a simple RISC-V computer system on FPGA. The FPAG board is Digilent Nexys A7-100T. The compter system is composed of: a CPU, a data memory, an instruction memory, a vga screen, a keyboard, and a PWM audio output. The CPU is a 5-stage pipeline processor, which can execute 32-bit RISC-V instructions. The system interface is a terminal, which can execute basic commands and run two software - Snake and Piano.

Keywords: RISC-V, FPGA, 5-stage pipeline, computer system

# Contents

| 1 | Introduction |                               |   |  |
|---|--------------|-------------------------------|---|--|
|   | 1.1          | Overall introduction          | 1 |  |
|   | 1.2          | FPGA                          | 1 |  |
|   | 1.3          | RISC-V                        | 2 |  |
| 2 | Hardware     |                               |   |  |
|   | 2.1          | 5-stage pipeline CPU          | 2 |  |
|   | 2.2          | Memory management             | 2 |  |
|   | 2.3          | Instruction and data memory   | 2 |  |
|   | 2.4          | VGA screen (Write-only)       | 2 |  |
|   | 2.5          | Keyboard (Read-only)          | 3 |  |
|   | 2.6          | PWM audio output (Write-only) | 4 |  |
|   | 2.7          | Timer (Read-only)             | 5 |  |
| 3 | Software     |                               |   |  |
|   | 3.1          | Terminal                      | 6 |  |
|   | 3.2          | Snake                         | 6 |  |
|   | 3 3          | Piano                         | 6 |  |

## 1 Introduction

#### 1.1 Overall introduction

In the realm of digital systems design, the creation of a fully functional computer system on a Field-Programmable Gate Array (FPGA) stands as a testament to the designer's knowledge of digital systems. This project centers around the development of a simple yet robust RISC-V computer system, implemented on the Digilent Nexys A7-100T FPGA board. At the heart of this project lies the CPU, which is a 5-stage pipeline processor that can execute 32-bit RISC-V instructions. The computer system designed for this project includes essential components such as a CPU, a data memory, an instruction memory, a VGA screen, a keyboard, and a PWM audio output. The user interacts with the system through a terminal, which supports several commands and can run two software - Snake and Piano.

This report is divided into two parts. The first segment delves into the hardware design, concentrating on the intricacies of the 5-stage pipeline CPU, memory management, and all other devices. The second part shifts focus to the software design, encompassing the implementation of the terminal, the engaging Snake game, and the musical pursuit in the form of the Piano game.

#### 1.2 FPGA

The FPGA board for this project is Digilent Nexys A7-100T, a circuit design and implementation platform for classroom use. For more information, please refer to https://digilent.com/reference/programmable-logic/nexys-a7/start.

#### 1.3 RISC-V

RISC-V is an open-source instruction set architecture (ISA) based on reduced instruction set computer (RISC) principles. It is a standard ISA designed to be simple, extensible, and easy to implement. In this project, I implemented a 32-bit RISC-V CPU, which can execute all 37 base instructions. For more information about RISC-V itself, please refer to https://riscv.org/.

## 2 Hardware

### 2.1 5-stage pipeline CPU

### 2.2 Memory management

The CPU uses byte addressing, and access all other devices through memory-mapped I/O. The memory address is 32 bits wide, and the first 12 bits are used to select the device. The address map is shown in Table 1.

# 2.3 Instruction and data memory

The instruction and data memory are both implemented using block RAM (BRAM). The instruction memory is read-only, and the data memory is read-write. Both of them are 1MB in size. The instruction address must be aligned to 4 bytes, while the data address can be any byte address.

# 2.4 VGA screen (Write-only)

The resolution is 640x480, and the color depth is 12 bits. Because the system is totally based on a terminal and the games are using characters as the basic

| Address range           | Device             |
|-------------------------|--------------------|
| 0x00000000 - 0x000FFFFF | Instruction memory |
| 0x00100000 - 0x001FFFFF | Data memory        |
| 0x00200000 - 0x002FFFFF | VGA screen         |
| 0x00300000, 0x00300004  | Keyboard           |
| 0x00400000, 0x00400004  | LED                |
| 0x00500000, 0x00500004  | timer              |
| 0x00600000              | deprecated         |
| 0x00700000              | deprecated         |
| 0x00800000, 0x00800004  | PWM audio output   |

Table 1: Address Map

unit, the screen is divided into 80x30 characters of 8x16 pixels. This allows us only store 80x30 characters' ascii code.

To accelerate the screen access speed, the screen is implemented with an one-dimensional array instead of a two-dimensional array. Considering the line and column size, 11-7 bits of the address are used to represent the line number, and 6-0 bits are used to represent the column number  $(2^5 = 32, 2^7 = 128)$ . The screen has a base address of 0x00200000, and is accessed with 0x00200000 + offset.

# 2.5 Keyboard (Read-only)

The input signals of the keyboard are PS2\_CLK and PS2\_DATA. With a keyboard signal processor, a byte of scan code can be generated when a key is pressed. The scan code is then stored in a circular buffer of 16 bytes. The buffer provides the CPU with a "new\_key" signal, which is high when

the buffer is not empty. The CPU can read "new\_key" through the address 0x00300000. When the CPU reads the address 0x00300004, the buffer will pop a byte of scan code.

The driver of the keyboard is implemented in the software part. Basically, the driver is a finite state machine (Figure 1), which can be divided into three states: KEY\_DOWN, KEY\_UP, LONG\_KEY\_DOWN, and LONG\_KEY\_UP. The "LONG\_" prefix in states refers to the two-bytes scan code which mainly start with 0xE0. The break code of a key is 0xF0 succeeded by its make code, thus the KEY\_UP and LONG\_KEY\_UP states are used to detect the break code.

The driver is implemented with a C function, which has only one state change every time it is called. There are three global values, "key\_ready", "two\_byte\_code" and "key\_up", which are signals for applications and are set according to the state. When the state machine finds that it ends a key press, it sets "key\_ready" to 1, and returns (the lowest byte of) the scan code. Otherwise, it returns 0.

The other two-byte scan codes are not used in this project, so we just ignore them.

# 2.6 PWM audio output (Write-only)

The audio pitch is based on the frequency of the PWM signal. There is a count value and a max value. The PWM signal is high when the count value is less than half of the max value, and low otherwise. The count value is increased by 1 every clock cycle and set back to 0 when it exceeds the max value. Thus, the pitch can be adjusted by changing the max value.

The audio has two memory addresses, 0x00800000 and 0x00800004. The first address is used to set the max value, and the second address is used to



Figure 1: Keyboard driver state machine

1: KEY\_DOWN, 2: LONG\_KEY\_DOWN,

3: KEY\_UP, 4: LONG\_KEY\_UP

turn on/off the audio.

# 2.7 Timer (Read-only)

There are two memory addresses, 0x00500000 and 0x00500004, corresponding to the millisecond counter and the second counter. Two extra counters are used for the timer. The first is a clock cycle counter that is set back to 0 when it exceeds 10<sup>5</sup>, and then both the second counter and the millisecond counter add 1. The second counter is set back to 0 when it exceeds 1000, and then the second counter adds 1. This implementation of four counters can largely reduce the complexity of the calculation because it doesn't use any division or multiplication.

- 3 Software
- 3.1 Terminal
- 3.2 Snake
- 3.3 Piano