Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 1 | --- |
| 2 | title: "Getting started on FPGAs" |
| 3 | --- |
| 4 | |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 5 | # Getting started on FPGAs |
| 6 | |
| 7 | Do you want to try out the lowRISC chip designs, but don't have a couple thousand or million dollars ready for an ASIC tapeout? |
| 8 | Running lowRISC designs on an FPGA board can be the answer! |
| 9 | |
| 10 | ## Prerequisites |
| 11 | |
| 12 | To use the lowRISC Comportable designs on an FPGA you need two things: |
| 13 | |
| 14 | * A supported FPGA board |
| 15 | * A tool from the FPGA vendor |
| 16 | |
| 17 | Depending on the design/target combination that you want to synthesize you will need different tools and boards. |
| 18 | Refer to the design documentation for information what exactly is needed. |
| 19 | |
Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 20 | * [Obtain an FPGA board]({{< relref "fpga_boards.md" >}}) |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 21 | |
Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 22 | Follow the install instructions to [prepare the system]({{< relref "install_instructions#system-preparation" >}}) and to install the [software development tools]({{< relref "install_instructions#software-development" >}}) and [Xilinx Vivado]({{< relref "install_instructions#xilinx-vivado" >}}). |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 23 | |
| 24 | ## Create an FPGA bitstream |
| 25 | |
| 26 | Synthesizing a design for a FPGA board is done with the following commands. |
| 27 | |
Timothy Chen | 25b0c66 | 2019-10-17 11:06:24 -0700 | [diff] [blame] | 28 | The FPGA build will pull in a program to act as the boot ROM. |
Greg Chadwick | 567cfd6 | 2019-10-31 16:15:33 +0000 | [diff] [blame] | 29 | This must be built before running the FPGA build. |
Miguel Osorio | ae67f6b | 2019-10-18 18:56:08 -0700 | [diff] [blame] | 30 | This is pulled in from the `sw/device/boot_rom` directory (see the `parameters:` section of the `hw/top_earlgrey/top_earlgrey_nexysvideo.core` file). |
Greg Chadwick | 567cfd6 | 2019-10-31 16:15:33 +0000 | [diff] [blame] | 31 | |
| 32 | To build it: |
| 33 | ```console |
| 34 | $ cd $REPO_TOP |
| 35 | $ make -C sw/device SW_DIR=boot_rom clean all |
| 36 | ``` |
| 37 | |
Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 38 | At the moment there is no check that the `rom.vmem` file is up to date, so it is best to follow the instructions to [Build software]({{< relref "getting_started_sw.md" >}}) and understand the FPGA's overall software flow. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 39 | |
| 40 | In the following example we synthesize the Earl Grey design for the Nexys Video board using Xilinx Vivado 2018.3. |
| 41 | |
| 42 | ```console |
| 43 | $ . /tools/xilinx/Vivado/2018.3/settings64.sh |
| 44 | $ cd $REPO_TOP |
Miguel Osorio | 39fc9a4 | 2019-10-31 15:05:40 -0700 | [diff] [blame] | 45 | $ make -C sw/device SW_DIR=boot_rom clean all |
Tom Roberts | 0ab9e7e | 2019-11-04 13:59:22 +0000 | [diff] [blame^] | 46 | $ fusesoc --cores-root . run --target=synth lowrisc:systems:top_earlgrey_nexysvideo |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 47 | ``` |
| 48 | |
| 49 | The resulting bitstream is located at `build/lowrisc_systems_top_earlgrey_nexysvideo_0.1/synth-vivado/lowrisc_systems_top_earlgrey_nexysvideo_0.1.bit`. |
Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 50 | See the [reference manual]({{< relref "ref_manual_fpga.md" >}}) for more information. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 51 | |
Tom Roberts | 0ab9e7e | 2019-11-04 13:59:22 +0000 | [diff] [blame^] | 52 | ## Connecting the board |
| 53 | |
| 54 | * Use a Micro USB cable to connect the PC with the *PROG*-labeled connector on the board. |
| 55 | * Use a second Micro USB cable to connect the PC with the *UART*-labled connector on the board. |
| 56 | * After connecting the UART, use `dmesg` to determine which serial port was assigned. It should be named `/dev/ttyUSB*`, e.g. `/dev/ttyUSB0`. |
| 57 | * Ensure that you have sufficient access permissions to the device, check `ls -l /dev/ttyUSB*`. The udev rules given in the Vivado installation instructions ensure this. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 58 | |
| 59 | ## Flash the bitstream onto the FPGA |
| 60 | |
| 61 | To flash the bitstream onto the FPGA you need to use either the Vivado GUI or the command line. |
| 62 | |
| 63 | ### Using the command line |
| 64 | |
| 65 | Use the following command to program the FPGA with fusesoc. |
| 66 | |
| 67 | ```console |
| 68 | $ . /tools/xilinx/Vivado/2018.3/settings64.sh |
| 69 | $ cd $REPO_TOP |
| 70 | $ fusesoc --cores-root . pgm lowrisc:systems:top_earlgrey_nexysvideo:0.1 |
| 71 | ``` |
| 72 | |
| 73 | Note: `fusesoc pgm` is broken for edalize versions up to (and including) v0.1.3. |
| 74 | You can check the version you're using with `pip3 show edalize`. |
Greg Chadwick | 567cfd6 | 2019-10-31 16:15:33 +0000 | [diff] [blame] | 75 | If you have having trouble with programming using the command line, try the GUI. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 76 | |
| 77 | ### Using the Vivado GUI |
| 78 | |
| 79 | ```console |
| 80 | $ . /tools/xilinx/Vivado/2018.3/settings64.sh |
| 81 | $ cd $REPO_TOP |
| 82 | $ make -C build/lowrisc_systems_top_earlgrey_nexysvideo_0.1/synth-vivado build-gui |
| 83 | ``` |
| 84 | |
| 85 | Now the Vivado GUI opens and loads the project. |
| 86 | |
| 87 | * Connect the FPGA board to the PC and turn it on. |
| 88 | * In the navigation on the left, click on *PROGRAM AND DEBUG* > *Open Hardware Manager* > *Open Target* > *Auto Connect*. |
| 89 | * Vivado now enumerates all boards and connects to it. (Note on Vivado 2018.1 you may get an error the first time and have to do auto connect twice.) |
| 90 | * Click on *Program Device* in the menu on the left (or at the top of the screen). |
Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 91 | * A dialog titled *Program Device* pops up. Select the file `lowrisc_systems_top_earlgrey_nexysvideo_0.1.bit` as *Bitstream file*, and leave the *Debug probes file* empty. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 92 | * Click on *Program* to flash the FPGA with the bitstream. |
| 93 | * The FPGA is ready as soon as the programming finishes. |
| 94 | |
| 95 | |
| 96 | ## Testing the demo design |
| 97 | |
Timothy Chen | d11f4b9 | 2019-10-21 12:53:45 -0700 | [diff] [blame] | 98 | The `hello_world` demo software shows off some capabilities of the design. |
Garret Kelly | 9eebde0 | 2019-10-22 15:36:49 -0400 | [diff] [blame] | 99 | In order to load `hello_world` into the FPGA, both the binary and the [loading tool]({{< relref "/sw/host/spiflash/README.md" >}}) must be compiled. |
Timothy Chen | d11f4b9 | 2019-10-21 12:53:45 -0700 | [diff] [blame] | 100 | Please follow the steps below. |
Timothy Chen | 25b0c66 | 2019-10-17 11:06:24 -0700 | [diff] [blame] | 101 | |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 102 | * Generate the bitstream and flash it to the FPGA as described above. |
| 103 | * Open a serial console (use the device file determined before) and connect. |
| 104 | Settings: 230400 baud, 8N1, no hardware or software flow control. |
| 105 | ```console |
Greg Chadwick | 567cfd6 | 2019-10-31 16:15:33 +0000 | [diff] [blame] | 106 | $ screen /dev/ttyUSB0 230400 |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 107 | ``` |
Philipp Wagner | d1fec0a | 2019-10-30 22:57:28 +0000 | [diff] [blame] | 108 | Note that the Nexsys Video demo program that comes installed on the board runs the UART at 115200 baud; |
| 109 | expect to see garbage characters if that is running. |
| 110 | This can happen if you connect the serial console before using Vivado to program your new bitstream or you press the *PROG* button that causes the FPGA to reprogram from the code in the on-board SPI flash. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 111 | * On the Nexys Video board, press the red button labeled *CPU_RESET*. |
Tom Roberts | 0ab9e7e | 2019-11-04 13:59:22 +0000 | [diff] [blame^] | 112 | * You should see the ROM code report its commit ID and build date. |
| 113 | * Run the loading tool. |
| 114 | ```console |
| 115 | $ cd ${REPO_TOP} |
| 116 | $ make -C sw/device SW_DIR=examples/hello_world SW_BUILD_DIR=out clean all |
| 117 | $ make -C sw/host/spiflash clean all |
| 118 | $ ./sw/host/spiflash/spiflash --input=sw/device/out/sw.bin |
| 119 | |
| 120 | Running SPI flash update. |
| 121 | Image divided into 6 frames. |
| 122 | frame: 0x00000000 to offset: 0x00000000 |
| 123 | frame: 0x00000001 to offset: 0x000003d8 |
| 124 | frame: 0x00000002 to offset: 0x000007b0 |
| 125 | frame: 0x00000003 to offset: 0x00000b88 |
| 126 | frame: 0x00000004 to offset: 0x00000f60 |
| 127 | frame: 0x80000005 to offset: 0x00001338 |
| 128 | ``` |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 129 | * Observe the output both on the board and the serial console. Type any text into the console window. |
| 130 | * Exit `screen` by pressing CTRL-a k, and confirm with y. |
| 131 | |
| 132 | ## Develop with the Vivado GUI |
| 133 | |
| 134 | Sometimes it is helpful to use the Vivado GUI to debug a design. |
| 135 | fusesoc makes that easy, with one small caveat: by default fusesoc copies all source files into a staging directory before the synthesis process starts. |
| 136 | This behavior is helpful to create reproducible builds and avoids Vivado modifying checked-in source files. |
| 137 | But during debugging this behavior is not helpful. |
| 138 | The `--no-export` option of fusesoc disables copying the source files into the staging area, and `--setup` instructs fusesoc to only create a project file, but not to run the synthesis process. |
| 139 | |
| 140 | ```console |
| 141 | $ # only create Vivado project file |
| 142 | $ fusesoc --cores-root . build --no-export --setup lowrisc:systems:top_earlgrey_nexysvideo |
| 143 | ``` |
| 144 | |
| 145 | ## Connect with OpenOCD and debug |
| 146 | |
| 147 | To connect the FPGA with OpenOCD, run the following command |
| 148 | |
| 149 | ```console |
| 150 | $ cd $REPO_TOP |
| 151 | $ /tools/openocd/bin/openocd -s util/openocd -f board/lowrisc-earlgrey-nexysvideo.cfg |
| 152 | ``` |
| 153 | |
| 154 | To actually debug through OpenOCD, it must either be connected through telnet or GDB. |
| 155 | |
| 156 | ### Debug with OpenOCD |
| 157 | |
| 158 | The following is an example for using telnet |
| 159 | |
| 160 | ```console |
| 161 | $ telnet localhost 4444 // or whatever port that is specificed by the openocd command above |
| 162 | $ mdw 0x8000 0x10 // read 16 bytes at address 0x8000 |
| 163 | ``` |
| 164 | |
| 165 | ### Debug with GDB |
| 166 | |
| 167 | An example connection with GDB, which prints the registers after the connection to OpenOCD is established |
| 168 | |
| 169 | ```console |
| 170 | $ cd $REPO_TOP |
Tom Roberts | 0ab9e7e | 2019-11-04 13:59:22 +0000 | [diff] [blame^] | 171 | $ /tools/riscv/bin/riscv32-unknown-elf-gdb -ex "target extended-remote :3333" -ex "info reg" sw/device/boot_rom/rom.elf |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 172 | ``` |
| 173 | |
Greg Chadwick | 567cfd6 | 2019-10-31 16:15:33 +0000 | [diff] [blame] | 174 | Note that debug support is not yet mature (see https://github.com/lowRISC/opentitan/issues/574). |
| 175 | In particular GDB cannot set breakpoints as it can't write to the (emulated) flash memory. |
| 176 | HW breakpoint support is planned for Ibex to allow breakpointing code in flash. |
| 177 | |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 178 | #### Common operations with GDB |
| 179 | |
| 180 | Examine 16 memory words in the hex format starting at 0x200005c0 |
| 181 | |
| 182 | ```console |
| 183 | (gdb) x/16xw 0x200005c0 |
| 184 | ``` |
| 185 | |
| 186 | Press enter again to print the next 16 words. |
| 187 | Use `help x` to get a description of the command. |
| 188 | |
| 189 | If the memory content contains program text it can be disassembled |
| 190 | |
| 191 | ```console |
| 192 | (gdb) disassemble 0x200005c0,0x200005c0+16*4 |
| 193 | ``` |
| 194 | |
| 195 | Displaying the memory content can also be delegated to OpenOCD |
| 196 | |
| 197 | ```console |
| 198 | (gdb) monitor mdw 0x200005c0 16 |
| 199 | ``` |
| 200 | |
| 201 | Use `monitor help` to get a list of supported commands. |
| 202 | |
Greg Chadwick | 567cfd6 | 2019-10-31 16:15:33 +0000 | [diff] [blame] | 203 | To single-step use `stepi` or `step` |
| 204 | |
| 205 | ```console |
| 206 | (gdb) stepi |
| 207 | ``` |
| 208 | |
| 209 | `stepi` single-steps an instruction, `step` single-steps a line of source code. |
| 210 | When testing debugging against the hello_world binary it is likely you will break into a delay loop. |
| 211 | Here the `step` command will seem to hang as it will attempt to step over the whole delay loop with a sequence of single-step instructions which may take quite some time! |
| 212 | |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 213 | To change the program which is debugged the `file` command can be used. |
| 214 | This will update the symbols which are used to get information about the program. |
Philipp Wagner | 544ac9b | 2019-10-30 22:56:32 +0000 | [diff] [blame] | 215 | It is especially useful in the context of our `rom.elf`, which resides in the ROM region, which will eventually jump to a different executable as part of the flash region. |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 216 | |
| 217 | ```console |
Philipp Wagner | 544ac9b | 2019-10-30 22:56:32 +0000 | [diff] [blame] | 218 | (gdb) file sw/device/examples/hello_world/sw.elf |
lowRISC Contributors | 802543a | 2019-08-31 12:12:56 +0100 | [diff] [blame] | 219 | (gdb) disassemble 0x200005c0,0x200005c0+16*4 |
| 220 | ``` |
| 221 | |
| 222 | The output of the disassemble should now contain additional information. |