# FPGA-ZCU Quick Start Tutorial

Document Version 2.0c

Tuo Li, Sri Parameswaran School of CSE, University of New South Wales tuoli@unsw.edu.au, sri.parameswaran@unsw.edu.au May 5, 2021

# **About This Tutorial**

This document will provide a step-by-step walkthrough with the (rc-)fpga-zcu (the ZCU102 port of Rocket-chip) project, towards running operating system on rocket-chip System-on-Chip (SoC) on ZCU102 FPGA board. This tutorial assumes using SD card to boot and program the FPGA. This tutorial assumes that Vivado 2017.1 and SDK 2018.2 have been installed. If you meet any problems, you might find the solution in Section 6.

## 1 Getting Started

- Clone the main repository using git command from github: git clone https://github.com/li3tuo4/rc-fpga-zcu.git Alternatively, you can directly download the zip file from github. Let's denote the root directory of the main repo as [root-repo-dir].
- 2. Enter the [root-repo-dir]/zcu102 directory and use Makefile to initiate the submodules/sub-repositories:

  make init-submodules
- 3. Enter [root-repo-dir]/rocket-chip directory and follow README to install the rocket-chip dependencies (submodules). Make sure riscv-linux-gnu is installed when building compiler. This is required for making riscv-linux.

### 2 Hardware Generation

- 1. Enter [root-repo-dir]/zcu102 directory and use Makefile to compile chisel source code: make rocket
- 2. Implement and build bitstream using Makefile: make bitstream
- 3. Export the hardware from the vivado workspace (the directory created during synthesis and implementation):
  - cp zcu102\_rocketchip\_ZynqConfig/zcu102\_rocketchip\_ZynqConfig.runs/impl\_1/
     rocketchip\_wrapper.sysdef soft\_config/rocketchip\_wrapper.hdf
    cp zcu102\_rocketchip\_ZynqConfig/zcu102\_rocketchip\_ZynqConfig.runs/impl\_1/
     rocketchip\_wrapper.bit soft\_config/rocketchip\_wrapper.bit

These files are used by software generation.

### 3 Software Generation

In this step, we will create the software stacks for ARM and RISC-V. The ARM operating system will run on the host (ARM cores), which calls RISC-V via frontend server (fesvr). Executing frontend server with Berkeley boot loader (bbl), built with RISC-V Linux kernel, on ARM, will boot up RISC-V core to run RISC-V Linux operating system.

NB! If you only want to use riscv-pk (a lightweight proxy kernel) with fesvr, Step 1 to 5 in Section 3.2 can be ignored. And, in Step 6 and 7, make and put pk into rootfs, instead of bbl.

### 3.1 ARM

1. Install petalinux, which creates software stacks for ARM. Follow the "Installation Steps" on Page 11 in: https://www.xilinx.com/support/documentation/sw\_manuals/xilinx2017\_

### 1/ug1144-petalinux-tools-reference-guide.pdf

- 2. Download board support package (bsp) file of ZCU102 from: https://www.xilinx.com/member/forms/download/xef.html?filename=xilinx-zcu102-2017.1-final.bsp. Xilinx account is required, which can be registered straightforwardly. Put the downloaded file xilinx-zcu102-2017.1-final.bsp into [root-repo-dir]/zcu102/soft\_config directory.
- 3. Export PETALINUX environment variable and set up petalinux by: export PETALINUX=/path/to/petalinux && make pl\_setup
- 4. Enter [root-repo-dir]/zcu102. Use Makefile to automatically create and build petalinux project for ARM Linux image:

make kernel\_image

Makefile will automatically create a petalinux project named petalinux\_proj, in [root-repo-dir]/zcu102. The BOOT.BIN and image.ub in petalinux\_proj/images/linux are the image files created. initramfs.cpio.gz is the root file system.

5. Copy these three files to [root-repo-dir]/zcu102/fpga-images-zcu102 directory. Note this step will replace the initial image files in this directory. Note you can use Makefile to open, modify, and close the root filesystem:

make rootfs-open
[modify]
make rootfs-close

#### 3.2 RISC-V

1. Make sure Xilinx SDK is installed and has been added to the PATH to successfully generate the binary. Enter the zcu102 directory and use Makefile to create fesvr-zynq binary for ZCU102:

cd [root-repo-dir]/zcu102
make fesvr-zynq

The target files including fesur-zynq and libfesur.so will be generated in [root-repo-dir]/common/build.

- 2. Obtain freedom-u-sdk from <a href="https://github.com/sifive/freedom-u-sdk">https://github.com/sifive/freedom-u-sdk</a>. Make sure the system environment is set up as specified by README of the repo. Note you might need to unset RISCV environment variable for building riscv-linux (you can set it again after build). This tutorial uses linux\_u500vc707devkit\_confg branch (commit db77bd7a8779d776d3f67b8a76ab5973e93604e6).
- 3. Enter the freedom-u-sdk directory and initialize the repository: cd /path/to/freedom-u-sdk

git submodule update --recursive --init

- 4. Copy the configuration file from [root-repo-dir]/zcu102/soft\_config to the freedom-u-sdk (optionally you can backup the default config file at first):
  - cp /path/to/freedom-u-sdk/conf/linux\_defconfig /path/to/freedom-u-sdk/conf/
     linux\_defconfig\_bak
  - cp /path/to/soft\_config/config\_freedom /path/to/freedom-u-sdk/conf/linux\_defconfig

5. In freedom-u-sdk root directory, make riscv-linux kernel: make vmlinux

The kernel image vmlinux will be generated in /path/to/freedom-u-sdk/work/linux.

6. Follow the riscv-tools README to create Berkeley boot loader (bbl) with RISC-V Linux included:

```
cd [root-repo-dir]/rocket-chip/riscv-tools/riscv-pk/build
rm -rf *
../configure --prefix=$RISCV --host=riscv64-unknown-linux-gnu
--with-payload=/path/to/freedom-u-sdk/work/linux/vmlinux
make
make install
```

- 7. Put fesvr-zynq, libfesvr.so and bbl into the root filesystem along with the commands introduced in Step 5 in Section 3.1. After you opened the initramfs (rootfs), copy the files into /path/to/rootfs (sudo probably is needed):
  - cp /path/to/riscv-pk/build/bbl [root-repo-dir]/common/build/fesvr\_zynq [root-repodir]/common/build/ /path/to/rootfs/home/root/

After you finish copying, close the rootfs.

## 4 FPGA Board Setup and SD Card Preparation

- Prepare the board following Xilinx quick start guide in https://www.xilinx.com/support/documentation/boards\_and\_kits/zcu102/xtp426-zcu102-quickstart.pdf and Chapter
   Board Setup and Configuration in Xilinx UG1182 in https://www.xilinx.com/support/documentation/boards\_and\_kits/zcu102/ug1182-zcu102-eval-bd.pdf. Note Switch SW6 must be configured to "off, off, off, on" for allowing SD card boot up.
- 2. Prepare SD card following README in the sub-repository [root-repo-dir]/zcu102/fpga-images-zcu102 in https://github.com/li3tuo4/fpga-images-zcu.
- Enter [root-repo-dir]/zcu102 directory and use Makefile to load SD card: make load-sd Now it's ready to run the system on FPGA.

### 5 FPGA Emulation

Make sure USB cable's driver is installed properly on the host PC. Xilinx UG344 might be helpful: https://www.xilinx.com/support/documentation/user\_guides/ug344.pdf

Set up and open minicom or your preferred terminal: minicom -D /dev/ttyUSB0
 Make sure the serial port in minicom is configured to "115200 8N1", which means "speed = 115200, parity bit = none, data bits = 8".

- 2. Insert the SD card and power on the FPGA board. The minicom terminal will show the ARM Linux boot up output. By default, ARM Linux's login username and password are both "root".
- 3. After login, execute fesvr with bbl as argument to wake up RISC-V and let it boot up Linux: LD\_LIBRARY\_PATH=./ ./fesvr-zynq bbl
  By default, the login username and password of RISC-V Linux are "root" and "sifive".

Hint: If you have ethernet connection, which is already available in the default system build, using SCP to transfer files such as fesvr and bbl directly from Host PC to FPGA after ARM Linux login can save the time in opening, modifying and closing root filesystem.

### 6 Possible Build Errors and Solutions

### 6.1 Error compiling sbt component 'compiler-interface'

```
Example error message after make rocket:
sajid2@sajid2-HP-Compaq-Elite-8300-SFF:~/fpga-zynq/zc706$ make rocket
mkdir -p /home/sajid2/fpga-zynq/common/build
```

cd /home/sajid2/fpga-zynq/common && java -Xmx2G -Xss8M -XX:MaxPermSize=256M -jar /home/
sajid2/fpga-zynq/rocket-chip/sbt-launch.jar "run /home/sajid2/fpga-zynq/common/build
zyng Top zyng ZyngFPGAConfig"

Java HotSpot(TM) 64-Bit Server VM warning: ignoring option MaxPermSize=256M; support was removed in 8.0

```
[ERROR] Failed to construct terminal; falling back to unsupported java.lang.NumberFormatException: For input string: "0x100"
```

at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)

. .

[info] Resolving org.scala-sbt#interface; 0.13.16 ...

[error] (compile:compileIncremental) Error compiling sbt component 'compiler-interface' Project loading failed: (r)etry, (q)uit, (l)ast, or (i)gnore? q

### Solution:

- 1. Build/compile the chisel and verilator according to the instruction given here <a href="https://github.com/freechipsproject/chisel3">https://github.com/freechipsproject/chisel3</a> including publish\_local and adding librarydependencies into build.sbt file which locates into rocketchip directory. According to the following instructions:
  - (a) To publish your version of Chisel to the local Ivy (sbt's dependency manager) repository, run: sbt publishLocal

(b) In order to have your projects use this version of Chisel, you should update the library-Dependencies setting in your project's build.sbt file to:

```
libraryDependencies += "edu.berkeley.cs" %% "chisel3" % "3.2-SNAPSHOT"
```

- 2. Install Java 8.
  - (a) Downladed the java version 8 from https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html
  - (b) Extract to the folder
  - (c) Add the extracted folder bin path into path by

```
export ...../bin :$PATH
```

Check the java version before adding the path and after adding the path to confirm it. Use command which java:

```
sajid2@sajid2-HP-Compaq-Elite-8300-SFF: ~/rc-fpga-zcu/rocket-chip/riscv-tools$
    which java
/usr/bin/java
sajid2@sajid2-HP-Compaq-Elite-8300-SFF: ~/rc-fpga-zcu/rocket-chip/riscv-tools$
    export PATH=/home/sajid2/jdk-8u191-linux-x64/jdk1.8.0_191/bin:$PATH
sajid2@sajid2-HP-Compaq-Elite-8300-SFF: ~/rc-fpga-zcu/rocket-chip/riscv-tools$
    which java
/home/sajid2/jdk-8u191-linux-x64/jdk1.8.0_191/bin/java
```

3. Now make rocket. Finally it is working now and successfully completing

### 6.2 tcmalloc failure during vivado synthesis

```
Error message:

Start Technology Mapping

src/central_freelist.cc:333] tcmalloc: allocation failed 16384

TclStackFree: incorrect freePtr. Call out of sequence?

[Tue Jan 15 17:23:56 2019] synth_1 finished

wait_on_run: Time (s): cpu = 00:00:00.33 ; elapsed = 00:20:41 . Memory (MB): peak = 1493.863 ; gain = 0.000 ; free physical = 1893 ; free virtual = 3117

# launch_runs impl_1 -to_step write_bitstream

ERROR: [Common 17-70] Application Exception: Failed to launch run 'impl_1' due to failures in the following run(s):

synth_1

These failed run(s) need to be reset prior to launching 'impl_1' again.
```

#### Solution:

It's caused by insufficient memory in the host machine. According to https://www.xilinx.com/products/design-tools/vivado/memory.html, synthesizing for xczu9eg (zcu102) by vivado requires 10 to 14 GB memory.