This repository provides a minimal hardware-based demonstration of GPUDirect RDMA. This feature allows a PCIe device to directly access CUDA memory, thus allowing zero-copy sharing of data between CUDA and a PCIe device.
The code supports:
- NVIDIA Jetson AGX Xavier (Jetson) running Linux for Tegra (L4T).
- NVIDIA Drive AGX Xavier running Embedded Linux.
- A PC running the NVIDIA CUDA drivers and containing a Quadro or Tesla GPU.
A graphical representation of the system configuration created by the software in this repository, and the data flow between components, is shown below:
This project uses an FPGA as the PCIe device which accesses CUDA memory. The following FPGA boards are supported:
- RHS Research PicoEVB.
- HiTech Global HTG-K800.
The following sections detail how to obtain and program each board.
PicoEVB is an M.2 form-factor FPGA board which attaches to the host's PCIe bus for application data transfer, and is programmed via the M.2 connector's USB bus. It is available from:
- picoevb.com.
- Amazon; search for ASIN "B0779PC8S4" or "PicoEVB".
The PicoEVB board is a double-sided M.2 device. Jetson physically only supports boards with a full-size PCIe connector, or single-sided M.2 devices. PCs typically only support boards with a full-size PCIe connector. Some form of adapter is required to connect the two in a mechanically reliable way.
A PCIe x16/x8/x4/x2/x1 to M.2 key E adapter may be used to plug the PicoEVB board into a full-size PCIe slot on Jetson or a PC. One such adapter board may be available from Amazon as ASIN B013U4401W, product name "Sourcingbay M.2(NGFF) Wireless Card to PCI-e 1X Adapter".
The following pair of adapters may be used to connect the PicoEVB board to Jetson's M.2 key E connector:
- M.2 2230 key E to Mini-PCIe adapter with cable. This may be available from Amazon as ASIN B07JFYSNVL, product name "M.2 (NGFF) Key A/E/A+E to Mini PCI-E Adapter with FFC Cable". Alternatively, this may be available from Amazon as ASIN B00JSBPF70, product name "Bplus: P15S-P15F, M.2 (NGFF) to mPCIe Extender Board".
- Mini-PCIe to M.2 2230 adapter board. This may be available from Amazon as ASIN B07D4FCD1K, product name "HLT M.2 (NGFF) to mPCIe (PCIe+USB) Adapter".
The following adapter may be used to connect the PicoEVB board to Drive AGX Xavier's M.2 key M connector:
- M.2 KEY-M interface to M.2 KEY A-E or KEY E interface adapter. This may be available from AliExpress with product name as "M.2 Wifi Adapter M.2 M Key to A+E Key Slot Wifi Bluetooth Network Card NGFF NVMe PCI express SSD Port to E key Slot Wifi Adapter".
The set of available adapters and vendors is very variable over time. Some searching may be required to locate suitable adapters, from either Amazon or alternative websites.
HTG-K800 is a full-size x16 PCIe card. This will fit directly into the full- size PCIe connector on Jetson or a desktop PC. For more information, see: http://www.hitechglobal.com/Boards/Kintex-UltraScale.htm
This project supports the XCKU-60 FPGA, although this should be easy to change simply by changing the FPGA project properties and re-synthesizing the provided Vivado project.
You will need a Xilinx Platform Cable USB II to program the FPGA. For more information, see: https://www.xilinx.com/products/boards-and-kits/hw-usb-ii-g.html
This software must run on an x86 Linux PC.
Xilinx Vivado is used to compile the FPGA bitstream, and to program the bitstream into the FPGA. The free "WebPACK Edition" is sufficient. Obtain this software from the Xilinx website.
The PicoEVB project requires Vivado 2018.3.
The HTG-K800 project requires Vivado 2018.1.
Newer versions of Vivado should be able to import these projects.
This software is only required for the PicoEVB board; the HTG-K800 board does not need it.
This software must run on the system that the PicoEVB FPGA card is plugged into. This may be either an x86 Linux PC, or a Jetson system.
Note that Drive AGX Xavier's M.2 slot does support USB. Consequently, it is not possible to program the FPGA when attached to this connector. The PicoEVB board can be programmed on Linux PC or Jetson system instead, then connected to Drive AGX Xavier once programmed.
Vivado relies upon a piece of software known as xvcd (Xilinx Virtual Cable Daemon) to communicate with the PicoEVB board for programming purposes. Obtain it from github.com. Execute the commands below to download and compile the software:
sudo apt update
sudo apt install build-essential libftdi-dev
git clone https://github.com/RHSResearchLLC/xvcd.git
cd xvcd/
cd linux/src
make
In the following text, fpga-*/
refers to the FPGA project sub-directory. For
PicoEVB, this is fpga-picoevb/
, and for the HTG-K800, this is
fpga-htg-k800/
.
A pre-compiled bitstream is provided in this project; fpga-*/*.mcs.bz2
. It is
not necessary to regenerate the bitstream. However, if you wish to do so,
follow these steps:
- Open a shell prompt, and
cd
to thefpga-*/
directory in this project. - Execute
./git-to-project.sh
to generate the Vivado project files. You may have to adjust thevivado
variable in this script if thevivado
executable is not in your$PATH
, or the expected installation location. - Execute
./synthesize-fpga.sh
to synthesize and implement the FPGA. This will generate the FPGA bitstream. Alternatively, you may perform this step by openingfpga-*/vivado-project/vivado-project.xpr
using the Vivado GUI, and requesting that it perform bitstream generation. Either way, this process will take from 5-60 minutes depending on the speed of your PC, and which FPGA project you're building. - Execute
./generate-cfgmem.sh
to generate the configuration memory image.
If you make modifications to the Vivado project, or any files or IP blocks it
contains or uses, and wish to commit those changes into source control, execute
./project-to-git.sh
to regenerate the checked-in files git-to-project.tcl
and git-to-ips.tcl
.
Programming the FPGA requires Vivado installed on an x86 Linux PC, and xvcd running on the system that contains the PicoEVB board.
If you run xvcd on Jetson, you must allow network connections from Vivado on
your x86 Linux PC to xvcd running on Jetson. The simplest way to do this is to
use ssh
's port-forwarding feature; on the x86 Linux PC, execute:
ssh -L 2542:127.0.0.1:2542 ip_address_of_jetson
To run xvcd, on the system containing the FPGA card, execute:
sudo ./xvcd -P 0x6015
On your x86 Linux PC, open a shell prompt, cd
to the fpga-*/
directory in
this project, and execute:
program-fpga.sh
The process of connecting Vivado's programming tools to the FPGA can be unreliable. If the connection attempt fails, and the script exits without programming the FPGA, you will need to execute the command again.
The programming process will take from 20 to 40 minutes. The programming process generates no output for most of its operation, so may appear to have hung, but is actually running.
Programming the FPGA requires Vivado installed on an x86 Linux PC with the Xilinx platform cable attached.
On your x86 Linux PC, open a shell prompt, cd
to the fpga-*/
directory in
this project, and execute:
program-fpga.sh
The programming process will take a few minutes. The programming process generates no output for most of its operation, so may appear to have hung, but is actually running.
Please use rel-36+ branch for kernel 5.9+ version.
To build the Linux kernel driver, execute:
sudo apt update
sudo apt install build-essential bc
cd /path/to/this/project/kernel-module/
./build-for-jetson-drive-igpu-native.sh
This will generate picoevb-rdma.ko
.
The Linux kernel driver may alternatively be built (cross-compiled) on an x86
Linux PC. You will first need to obtain a copy of the "Linux headers" or
"kernel external module build tree" files from L4T; these may be found in
/usr/src/
on Jetson/Drive AGX Xavier, or obtained from the L4T/Drive
downloads website.
To build the Linux kernel driver on a x86 Linux PC, execute:
sudo apt update
sudo apt install build-essential bc
cd /path/to/this/project/kernel-module/
# Adjust the KDIR value to match the exact path in your copy of the
# kernel headers
KDIR=/path/to/linux-headers-4.9.140-tegra-linux_x86_64/kernel-4.9/ ./build-for-jetson-drive-igpu-on-pc.sh
This will generate picoevb-rdma.ko
. This file must be copied to Jetson/Drive
AGX Xavier.
sudo apt update
sudo apt install build-essential bc
cd /path/to/this/project/kernel-module/
./build-for-pc-native.sh
This will generate picoevb-rdma.ko
.
To load the kernel module, execute:
sudo insmod ./picoevb-rdma.ko
Once the module is loaded, executing lspci -v
should show that the module is
in use as the kernel driver for the FPGA board:
$ lspci -v
...
0003:01:00.0 Memory controller: NVIDIA Corporation Device 0001
Subsystem: NVIDIA Corporation Device 0001
Flags: bus master, fast devsel, latency 0, IRQ 36
Memory at 34210000 (32-bit, non-prefetchable) [size=4K]
Memory at 34200000 (32-bit, non-prefetchable) [size=64K]
Capabilities: <access denied>
Kernel driver in use: picoevb-rdma
The client applications are best built on Jetson/Drive AGX Xavier itself. Make sure you have the CUDA development tools installed, and execute:
sudo apt update
sudo apt install build-essential bc
cd /path/to/this/project/client-applications/
./build-for-jetson-drive-igpu-native.sh
Building (cross-compiling) the client applications on a x86 Linux PC is only partially supported; the makefile does not yet support cross-compiling the CUDA test application. However, the other applications may be cross-compiled by executing:
sudo apt update
sudo apt install build-essential bc
cd /path/to/this/project/client-applications/
./build-for-jetson-drive-igpu-on-pc.sh
You may need to adjust the value of variable CROSS_COMPILE
in script
./build-for-jetson-drive-igpu-on-pc.sh
to match the configuration of your x86
Linux PC.
Make sure you have the CUDA development tools installed, and execute:
sudo apt update
sudo apt install build-essential bc
cd /path/to/this/project/client-applications/
./build-for-pc-native.sh
Two PCIe data access tests are provided; rdma-malloc
and rdma-cuda
. Both
tests are structurally identical, but allocate memory using different APIs; the
former using malloc()
, and the latter via cudaHostAlloc()
(Jetson/Drive AGX
Xavier) or cudaMalloc()
(PC).
Both tests proceed as following:
- Allocate source and destination memory.
- In the CUDA case, prepare the memory for RDMA by calling
cuPointerSetAttribute(CU_POINTER_ATTRIBUTE_SYNC_MEMOPS)
and pinning it. - Fill the source surface with a known pattern.
- Fill the destination surface with different values.
- Use the FPGA to copy the source to the destination surface.
- Validate that the data was correctly copied.
To run the tests, execute:
sudo ./rdma-malloc
sudo ./rdma-cuda
You can avoid the need to use sudo
by applying appropriate permissions to the
kernel driver's device file, /dev/picoevb
.
Internally to the kernel driver, the copy operation divides the surface into 64KiB chunks (or smaller, depending on memory alignment), and for each chunk first copies that chunk's data from the source surface to the FPGA's internal memory, then copies the data from the FPGA's internal memory to the destination surface. This demonstrates both PCIe read and write access to CUDA GPU memory. The requirement to divide the data into chunks is a limitation of the internal memory size of the PicoEVB board's FPGA, and likely would not apply in a production device.
Separate test applications exist to exercise the uni-directional copy feature
of the kernel driver, and to report transfer performance. Two versions of the
tests exist; one using malloc()
'd memory on the host, and the other using
memory allocated via CUDA. To run these tests, execute:
sudo ./rdma-malloc-h2c-perf
sudo ./rdma-malloc-c2h-perf
sudo ./rdma-cuda-h2c-perf
sudo ./rdma-cuda-c2h-perf
This test sets the values of the three LEDs on the PicoEVB. It accepts a single
command-line parameter indicating the binary value to display on those LEDs.
The hardware inverts this value, so a parameter value of 0
turns on all LEDS,
and a parameter value of 7
turns off all LEDs. For example:
./set-leds 2
./set-leds 5