/jetson-containers

Running CUDA containers on the Jetson platform

Primary LanguageDockerfileMIT LicenseMIT

Jetson Containers

Running CUDA containers on the Jetson platform

  1. Preface
  2. Introduction
  3. Configuration
  4. Building
  5. Running
  6. OpenCV
  7. Flashing Devices
  8. Image Sizes

Preface

This project provides a repeatable, containerized approach to building software and libraries to run on the Nvidia Jetson platform.

Introduction

When building your application and choosing a base, the JetPack images can be trimmed to only include what is needed by your application. This can drastically reduce the size of your images.

These containers enable the OS can be flashed without JetPack having only the main driver pack installed. The other libraries will be included in the containers enabling migration and updates without involving the host operating system.

When building third party libraries, such as OpenCV and PyTorch, a swapfile will likely have to be created in the host OS. These packages require more memory than the system contains and will crash with very cryptic errors if they run out of memory.

The JetPack base images follow the Nvidia pattern of having base, runtime, and devel images for each device to start from. You cannot download the JetPack installers without logging in with the SDK manager, so the make tasks herein will automate the download/login through the SDK Manager and save off the device specific packages.

Note: if you have the SDK Manager open, the command line will not work. Nvidia has incorrectly handled their single instance implementation and the CLI will fail to run.

Configuration

The .env File

The Makefile scripts will import a .env file (for an example look at the .envtemp file) and export the variables defined.

  • REPO - This is for your own container registry/repo. The builds will take care of the tags and images names. For example REPO=mycontainers.azurecr.io/l4t
  • NV_USER - Sets the email address to be used when authenticating with the SDK Manager. This isn't stored and you'll be prompted for your password when the container runs.
  • DOCKER - Allows swapping out for another container runtime such as Moby or Balena. This variable is used in all container operations.
  • DOCKER_HOST - Setting the DOCKER_HOST variable will proxy builds to another machine such as a Jetson device. This allows running the make scripts from an x86_x64 host. This feature was added in November 2018. When using this feature, it is helpful to add your public key to the device's ~/.ssh/authorized_keys file. This will prevent credential checks on every build.
  • DOCKER_BUILD_ARGS - Allows adding arguments such as volume mounting or cleanup (-rm/--squash) during build operations.
  • DOCKER_RUN_ARGS - Allows adding arguments such as environment variables, mounts, network configuration, etc when running images. Can also be used to configure X11 forwarding.

Docker

Storing these images will also require significant disk space. It is highly recommended that an NVME or other hard drive is installed and mounted at boot through fstab. Once mounted, configure your container runtime to store its containers and images there. You'll also want to enable the experimental features to get --squash available during builds. This can be turned off by manually specifying DOCKER_BUILD_ARGS.

/etc/docker/daemon.json

{
    "data-root": "/some/external/docker",
    "experimental": true
}

Building

The project uses make to set up the dependent builds constructing the final images. The recipes fall into a few categories:

  • Driver packs (32.2, 32.1)
  • JetPack Dependencies (4.2.1, 4.2)
  • JetPack (4.2.1, 4.2)
  • Devices (jax (xavier), tx2, tx1, nano-dev)
  • Flashing containers
  • OpenCV (4.1.0)

Dependencies

The JetPack dependency builds must be run on an x86_64 host. They can be built with make <device>-jetpack-<jetpack-version>-deps. This will build an image using files downlaoded from the SDK Manager installed. Building this image will require authentication and NV_USER should be set in your .env file. As the container runs, enter your password when prompted. For sudo prompts, enter pass as the password. Once built, push the image to your container registry so that the device can leverage it all other builds. This may sound like a lot, but it is really just:

  • Set NV_USER in the .env file
  • Make sure DOCKER_HOST, if set, is pointing to an x86_64 host
  • make jax-jetpack-4.2-deps, or make nano-dev-jetpack-4.2-deps, or make tx2-jetpack-4.2-deps, or make jetpack-4.2-deps to build them all
  • Wait, then enter your Nvidia developer password when prompted
  • Upload finished image to your container registry

All other steps for JetPack require these previous steps to have been completed.

CTI Dependencies

CTI has a board support package which must be installed into the rootfs. When building flash images for CTI boards, there are three options:

  1. Run ~/jetson-containers$ make cti-<driver pack>-<device>-deps such as ~/jetson-containers$ make cti-32.1-tx2-deps which will download the BSP and generate an image l4t:cti-32.1-tx2-deps with the driver pack installed into /data.
  2. Download the BSP from the CTI web site and save it to an empty folder. In the .env file, set the SDKM_DOWNLOADS to this folder and run ~/jetson-containers$ make cti-<driver pack>-<device>-deps-from-folder. For example: ~/jetson-containers$ make cti-32.1-tx2-deps-from-folder which will generate an image l4t:cti-32.1-tx2-deps with the driver pack installed into /data.
  3. Download the BSP into the folder with the downloads from the SDK Manager. In the .env file, set the SDKM_DOWNLOADS to the chosen folder. Run ~/jetson-containers$ make <device>-jetpack-<version>-deps-from-folder. For example: ~/jetson-containers$ make tx2-jetpack-4.2-deps-from-folder which will generate an image l4t:tx2-jetpack-4.2-deps with the BSP and JetPack dependencies installed into /data.

For all of these, open your .env and set BSP_DEPENDENCIES_IMAGE to the dependency image created. In the last example it will be the same image as the DEPENDENCIES_IMAGE setting found in the /cti/<version>/drivers/*.conf file associated with the image being flashed.

Driver Packs

The driver packs form the base of the device images. Each version of JetPack is built on top of a driver pack. To build an image, follow the pattern:

make <driver-pack>-<device>-jetpack-<jetpack-version>

Note: not all combinations are valid and the Makefile should have all valid combinations declared. Note: if these command's are not run on the device, the DOCKER_HOST variable must be set. Note: from 4.2, building Jetson containers on the host is also supported: leave the DOCKER_HOST empty and make sure qemu-user-static is installed and interpreters are registered on the host. If not, please run:

sudo apt-get update && sudo apt-get install -y --no-install-recommends qemu-user-static binfmt-support

examples:

make driver-packs # build all driver pack bases
make jetpack-4.2.1 # build all JetPack 4.2.1. device builds and the driver packs they depend on
make 32.1-jax-jetpack-4.2
make 32.1-tx2-jetpack-4.2.1

There are additional recipes for building the 32.1-jax-jetpack-4.2 samples container (make build-32.1-jax-jetpack-4.2-samples) and running the container (make run-32.1-jax-jetpack-4.2-samples) which demonstrates mult-stage builds based on devel images.

Running

To run a container with GPU support there are two options. First, not recommended, run with the --privileged option. Second, recommended option, is to add the host's GPUs to the container:

docker run \
    --device=/dev/nvhost-ctrl \
    --device=/dev/nvhost-ctrl-gpu \
    --device=/dev/nvhost-prof-gpu \
    --device=/dev/nvmap \
    --device=/dev/nvhost-gpu \
    --device=/dev/nvhost-as-gpu \
    --device=/dev/nvhost-vic \
    <image-name>

X11 Forwarding

Running containerized applications with X11 forwarding requires simple configuration changes

  • On the host OS, run xhost +
  • Add --net=host and -e "DISPLAY" arguments to docker

The DOCKER_RUN_ARGS variable can be set in the .env file to DOCKER_RUN_ARGS=--net=host -e "DISPLAY" which will set up the forwarding when using make. Then run the run-32.1-jax-jetpack-4.2-samples task. This works for both local and remote sessions leveraging DOCKER_HOST. Note that when using DOCKER_HOST the forwarding will be done on the target and not forwarded locally.

OpenCV

Building OpenCV can take a few hours depending on your device and performance mode. When the OpenCV builds are complete, the install .sh file can be found in the /dist folder.

Building Custom OpenCV

make opencv-4.0.1-l4t-32.1-jetpack-4.2

Flashing Devices

Flashing devices usually requires installation of JetPack. To make this easier and more repeatable, you can now flash your device using a container.

There are default profiles which match the device and JetPack versions in the flash/jetpack-bases folder. Once the image is created, you have a versionable image which will be the base OS and drivers for your device.

Examples:

make image-jetpack # create all jetpack default configuration images for flashing.
make image-32.2-tx2i-jetpack-4.2.1 # JetPack 4.2.1 for TX2i with driver pack 32.2
make image-32.1-jax-jetpack-4.2 # JetPack 4.2 for Xavier with driver pack 32.1

Note: Ensure that if the DOCKER_HOST variable is set, the host specified must be x86_64. Note: The Dockerfiles can be modified to chroot the rootfs folder customizing the image such as setting up cloud-init.

To flash the device, put the device into recovery mode and connect it to the host via USB cable. You can follow the instructions from your device instructions, or run reboot recovery from a terminal session on the device. Now, run ./flash.sh along with the image tag you want to execute. For example, looking at images previously built:

>:~/jetson-containers/flash$ docker images
REPOSITORY                           TAG                 IMAGE ID            CREATED             SIZE
l4t:32.1-jax-jetpack-4.2-image     latest              e59950b4ebdc        5 hours ago         3.73GB

We can flash the Xavier with the default JetPack 4.2 configuration by executing ./flash.sh l4t:32.1-jax-jetpack-4.2-image. If your device is not in recovery mode, you will see an error similar to:

###############################################################################
# L4T BSP Information:
# R31 (release), REVISION: 1.0, GCID: 13194883, BOARD: t186ref, EABI: aarch64, 
# DATE: Wed Oct 31 22:26:16 UTC 2018
###############################################################################
Error: probing the target board failed.
       Make sure the target board is connected through 
       USB port and is in recovery mode.

If your device was in recovery mode, you should see progress displayed. Once the device has been flashed, it will automatically restart.

CTI BSPs Flashing

  1. Build BSP dependencies image
  2. Build JetPack dependencies image
  3. Build flashing image

Example: Building Orbitty

>:~/jetson-containers/$ make cti-32.1-tx2-125-deps
>:~/jetson-containers/$ make 32.1-tx2-jetpack-4.2-deps
>:~/jetson-containers/$ make image-cti-32.1-v125-orbitty
>:~/jetson-containers/$ ./flash/flash.sh l4t:cti-32.1-v125-orbitty-image

If you want to use a different dependency image for the BSP, set BSP_DEPENDENCIES_IMAGE in the .env file.

Image sizes

Base

Repository Tag Size
arm64v8/ubuntu bionic-20190912.1 57.7MB
arm64v8/ubuntu bionic-20190612 80.4MB
arm64v8/ubuntu xenial-20190904 110MB
arm64v8/ubuntu xenial-20190610 108MB

Jetson

Dependency packs:

Note that these are only used on build machines.

Repository Driver Size
l4t 32.2.1-host-jetpack-4.2.2-deps 2.6GB
l4t 32.2.1-jax-jetpack-4.2.2-deps 3.49GB
l4t 32.2.1-jax-8gb-jetpack-4.2.2-deps 3.49GB
l4t 32.2.1-tx2-jetpack-4.2.2-deps 3.49GB
l4t 32.2.1-tx2i-jetpack-4.2.2-deps 3.49GB
l4t 32.2.1-tx2-4gb-jetpack-4.2.2-deps 3.49GB
l4t 32.2.1-tx1-jetpack-4.2.2-deps 3.56GB
l4t 32.2.1-nano-dev-jetpack-4.2.2-deps 3.56GB
l4t 32.2.1-nano-jetpack-4.2.2-deps 3.56GB
l4t 32.2-host-jetpack-4.2.2-deps 2.58GB
l4t 32.2-jax-jetpack-4.2.1-deps 3.48GB
l4t 32.2-tx2-jetpack-4.2.1-deps 3.48GB
l4t 32.2-tx2i-jetpack-4.2.1-deps 3.48GB
l4t 32.2-tx2-4gb-jetpack-4.2.1-deps 3.48GB
l4t 32.2-tx1-jetpack-4.2.1-deps 3.55GB
l4t 32.2-nano-dev-jetpack-4.2.1-deps 3.55GB
l4t 32.2-nano-jetpack-4.2.1-deps 3.55GB
l4t 32.1-jax-jetpack-4.2-deps 3.30GB
l4t 32.1-tx2-jetpack-4.2-deps 3.30GB
l4t 32.1-nano-dev-jetpack-4.2-deps 3.29GB

Driver packs:

Repository Driver Size
l4t 32.2.1-jax 465MB
l4t 32.2.1-jax-8gb 465MB
l4t 32.2.1-tx2 465MB
l4t 32.2.1-tx2i 465MB
l4t 32.2.1-tx2-4gb 465MB
l4t 32.2.1-tx1 454MB
l4t 32.2.1-nano-dev 454MB
l4t 32.2.1-nano 454MB
l4t 32.2-jax 470MB
l4t 32.2-tx2 470MB
l4t 32.2-tx2i 470MB
l4t 32.2-tx2-4gb 470MB
l4t 32.2-tx1 460MB
l4t 32.2-nano-dev 460MB
l4t 32.2-nano 460MB
l4t 32.1-jax 479MB
l4t 32.1-nano-dev 469MB
l4t 32.1-tx2 479MB

JetPack 4.2.2

Repository Tag Size
l4t 32.2.1-jax-jetpack-4.2.2-base 475MB
l4t 32.2.1-jax-jetpack-4.2.2-runtime 1.23GB
l4t 32.2.1-jax-jetpack-4.2.2-devel 5.78GB
l4t 32.2.1-jax-8gb-jetpack-4.2.2-base 475MB
l4t 32.2.1-jax-8gb-jetpack-4.2.2-runtime 1.23GB
l4t 32.2.1-jax-8gb-jetpack-4.2.2-devel 5.78GB
l4t 32.2.1-tx2-jetpack-4.2.2-base 475MB
l4t 32.2.1-tx2-jetpack-4.2.2-runtime 1.23GB
l4t 32.2.1-tx2-jetpack-4.2.2-devel 5.78GB
l4t 32.2.1-tx2i-jetpack-4.2.2-base 475MB
l4t 32.2.1-tx2i-jetpack-4.2.2-runtime 1.23GB
l4t 32.2.1-tx2i-jetpack-4.2.2-devel 5.78GB
l4t 32.2.1-tx2-4gb-jetpack-4.2.2-base 475MB
l4t 32.2.1-tx2-4gb-jetpack-4.2.2-runtime 1.23GB
l4t 32.2.1-tx2-4gb-jetpack-4.2.2-devel 5.78GB
l4t 32.2.1-tx1-jetpack-4.2.2-base 464MB
l4t 32.2.1-tx1-jetpack-4.2.2-runtime 1.22GB
l4t 32.2.1-tx1-jetpack-4.2.2-devel 5.77GB
l4t 32.2.1-nano-jetpack-4.2.2-base 464MB
l4t 32.2.1-nano-jetpack-4.2.2-runtime 1.22GB
l4t 32.2.1-nano-jetpack-4.2.2-devel 5.77GB
l4t 32.2.1-nano-dev-jetpack-4.2.2-base 464MB
l4t 32.2.1-nano-dev-jetpack-4.2.2-runtime 1.22GB
l4t 32.2.1-nano-dev-jetpack-4.2.2-devel 5.77GB

JetPack 4.2.1

Repository Tag Size
l4t 32.2-jax-jetpack-4.2.1-base 480MB
l4t 32.2-jax-jetpack-4.2.1-runtime 1.23GB
l4t 32.2-jax-jetpack-4.2.1-devel 5.79GB
l4t 32.2-tx2-jetpack-4.2.1-base 480MB
l4t 32.2-tx2-jetpack-4.2.1-runtime 1.23GB
l4t 32.2-tx2-jetpack-4.2.1-devel 5.79GB
l4t 32.2-tx2i-jetpack-4.2.1-base 480MB
l4t 32.2-tx2i-jetpack-4.2.1-runtime 1.23GB
l4t 32.2-tx2i-jetpack-4.2.1-devel 5.79GB
l4t 32.2-tx2-4gb-jetpack-4.2.1-base 480MB
l4t 32.2-tx2-4gb-jetpack-4.2.1-runtime 1.23GB
l4t 32.2-tx2-4gb-jetpack-4.2.1-devel 5.79GB
l4t 32.2-tx1-jetpack-4.2.1-base 470MB
l4t 32.2-tx1-jetpack-4.2.1-runtime 1.22GB
l4t 32.2-tx1-jetpack-4.2.1-devel 5.78GB
l4t 32.2-nano-jetpack-4.2.1-base 470MB
l4t 32.2-nano-jetpack-4.2.1-runtime 1.22GB
l4t 32.2-nano-jetpack-4.2.1-devel 5.78GB
l4t 32.2-nano-dev-jetpack-4.2.1-base 470MB
l4t 32.2-nano-dev-jetpack-4.2.1-runtime 1.22GB
l4t 32.2-nano-dev-jetpack-4.2.1-devel 5.78GB

JetPack 4.2

Repository Tag Size
l4t 32.1-jax-jetpack-4.2-base 489MB
l4t 32.1-jax-jetpack-4.2-runtime 1.23GB
l4t 32.1-jax-jetpack-4.2-devel 5.69GB
l4t 32.1-jax-jetpack-4.2-samples 2.32GB
l4t 32.1-nano-dev-jetpack-4.2-base 479MB
l4t 32.1-nano-dev-jetpack-4.2-runtime 1.2GB
l4t 32.1-nano-dev-jetpack-4.2-devel 5.66GB
l4t 32.1-tx2-jetpack-4.2-base 489MB
l4t 32.1-tx2-jetpack-4.2-runtime 1.21GB
l4t 32.1-tx2-jetpack-4.2-devel 5.67GB