Skip to content

Latest commit

 

History

History
151 lines (97 loc) · 11.4 KB

build.md

File metadata and controls

151 lines (97 loc) · 11.4 KB

Building Containers

The jetson-containers build command is a proxy launcher for jetson_containers/build.py. It can be run from any working directory after you clone the repo and run the installer from the System Setup (you should also probably mount additional storage when building containers)

To list the packages available to build for your version of JetPack/L4T, you can use --list-packages and --show-packages:

$ jetson-containers build --list-packages       # list all packages
$ jetson-containers build --show-packages       # show all package metadata
$ jetson-containers build --show-packages ros*  # show all the ros packages

To build a container that includes one or more packages:

$ jetson-containers build pytorch               # build a container with just PyTorch
$ jetson-containers build pytorch jupyterlab    # build container with PyTorch and JupyterLab

The builder will chain together the Dockerfiles of each of packages specified, and use the result of one build stage as the base image of the next. The initial base image defaults to one built for your version of JetPack, but can be changed by specifying the --base argument.

The docker commands that get run during the build are printed and also saved to shell scripts under the jetson-containers/logs directory. To see just the commands it would have run without actually running them, use the --simulate flag.

Container Names

By default, the name of the container will be based on the package you chose to build. However, you can name it with the --name argument:

jetson-containers build --name=my_container pytorch jupyterlab

If you omit a tag from your name, then a default one will be assigned (based on the tag prefix/postfix of the package - typically $L4T_VERSION)

You can also build your container under a namespace ending in / (for example --name=my_builds/ would result in an image like my_builds/pytorch:r36.2.0). Using namespaces can help manage/organize your images if you are building a lot of them.

Multiple Containers

The --multiple flag builds separate containers for each of the packages specified:

$ jetson-containers build --multiple pytorch tensorflow2   # built a pytorch container and a tensorflow2 container
$ jetson-containers build --multiple ros*                  # build all ROS containers

If you wish to continue building other containers if one fails, use the --skip-errors flag (this only applies to building multiple containers)

Changing the Base Image

By default, the base container image used at the start of the build chain will be l4t-base on JetPack 4, l4t-jetpack on JetPack 5, and ubuntu:22.04 on JetPack 6. However, if you want to add packages to a container that you already have, you can specify your own base image:

jetson-containers build --base=my_container:latest --name=my_container:pytorch pytorch  # add pytorch to your container

Note

On JetPack 4/5, it's assumed that base images already have the JetPack components available/installed (CUDA Toolkit, cuDNN, TensorRT).

On JetPack 6 (L4T r36.x), the CUDA components will automatically be installed on top of your base image if required.

You can specify the CUDA version by using CUDA_VERSION environment variable, otherwise it will pick the CUDA version that was made available to the L4T version.

Changing Versions

Many packages are versioned, and define subpackages like pytorch:2.0, pytorch:2.1, ect in the package's configuration. For some core packages, you can control the default version of these with environment variables like PYTORCH_VERSION, PYTHON_VERSION, and CUDA_VERSION. Other packages referring to these will then use your desired versions instead of the previous ones:

CUDA_VERSION=12.4 jetson-containers build --name=cu124/ pytorch  # build PyTorch for CUDA 12.4

The dependencies are also able to specify with requires which versions of L4T, CUDA, and Python they need, so changing the CUDA version has cascading effects downstream and will also change the default version of cuDNN, TensorRT, and PyTorch (similar to how changing the PyTorch version also changes the default version of torchvision and torchaudio). The reverse also occurs in the other direction, for example changing the TensorRT version will change the default version of CUDA (unless you explicitly specify it otherwise).

Package Environment Variable Versions
cuda CUDA_VERSION cuda/config.py
cudnn CUDNN_VERSION cudnn/config.py
tensorrt TENSORRT_VERSION tensorrt/config.py
python PYTHON_VERSION python/config.py
pytorch PYTORCH_VERSION pytorch/config.py

Using these together, you can rebuild the container stack for the specific version combination that you want:

CUDA_VERSION=12.4 PYTHON_VERSION=3.11 PYTORCH_VERSION=2.3 \
  jetson-containers build l4t-text-generation

The available versions are defined in the package's configuration scripts (some you can simply add new releases to by referring to the new git tag/branch, others need pointed to specific release downloads). Not all combinations are compatible, as the versioned packages frequently define the minimum version of others in the build tree they rely on (for example, TensorRT 10 requires CUDA 12.4).

For packages that provide different versions but don't have their own environment variable defined, you can specify the desired version of them that your container depends on in the Dockerfile header under depends, and it will override the default and build using them instead (e.g. by using depends: onnxruntime:1.17 instead of depends: onnxruntime). Like above, these versions will also cascade across the build.

Note

On JetPack 6 (L4T r36.x), if you don't have CUDA_VERSION specified, it will pick the CUDA version that was packaged with that version of L4T in the JetPack.

  • L4T r36.2 (JetPack 6.0 DP) --> CUDA 12.2
  • L4T r36.3 (JetPack 6.0 GA) --> CUDA 12.4
  • L4T r36.4 (JetPack 6.1) --> CUDA 12.6

Pip Server

Being able to change the versions of CUDA, Python, ect in the build tree will cause all the dependant packages like PyTorch to need rebuilt, and this can be time-consuming to recompile everything from scratch. In order to cache and re-use the Python wheels compiled during the container builds, there's a pip server running that wheels are uploaded to. The containers automatically use this pip server via the PIP_INDEX_URL environment variable that gets set in the base containers, and there are indexes created for different versions of JetPack/CUDA.

Then when the containers are built, it will first attempt to install the wheel from the server, and if it's found it builds it from source. There is another similar server run for caching tarball releases for projects that install C/C++ binaries and headers or other general files outside of pip. See this forum post for more information about the caching infrastructure, and the ability to use the pip server outside of just containers.

Building External Packages

Let's say that you have a project that you want to build a container for - if you define a custom package (i.e. by creating a Dockerfile with a YAML header), you can point build.sh to out-of-tree locations using the --package-dirs option:

jetson-containers build --package-dirs=/path/to/your/package your_package_name

You can also add jetson-containers as a git submodule to your project and build it that way (see jetson-inference as an example of this)

Tests

By default, tests are run during container builds for the packages that provide test scripts. After each stage of the build chain is complete, that package will be tested. After the build is complete, the final container will be tested against all of the packages again. This is to assure that package versions aren't inadvertantly changed/overwritten/uninstalled at some point during the build chain (typically, the GPU-accelerated version of the package being supplanted by the one that some other package installs from pip/apt/ect)

You can skip these tests however with the --skip-tests argument, which is a comma/colon-separated list of package names to skip their tests. Or all or * will skip all tests, while intermediate will only run tests at the end (not during).

$ jetson-containers build --skip-tests=numpy,onnx pytorch    # skip testing the numpy and onnx packages when building pytorch
$ jetson-containers build --skip-tests=all pytorch           # skip all tests
$ jetson-containers build --skip-tests=intermediate pytorch  # only run tests at the end of the container build

These flags tend to get used more during development - normally it's good to thoroughly test the build stack, as to not run into cryptic issues with packages later on.

Running Containers

To launch containers that you've built or pulled, see the Running Containers documentation or the package's readme page.

Troubleshooting

When building a container, you may hit this GitHub API rate limit especially if you are working in an office environment or similar where your outward-facing IP address is shared with other developers/instances.

ADD failed: failed to GET https://api.github.com/repos/abetlen/llama-cpp-python/git/refs/heads/main with status 403 Forbidden: {"message":"API rate limit exceeded for 216.228.112.22...

If that is the case, use --no-github-api option when running build.sh

$ jetson-containers build --no-github-api --skip-test=all text-generation-webui

The option --no-github-api is to remove a line like below from Dockerfile that was added to force rebuild on new git commits.

ADD https://api.github.com/repos/${LLAMA_CPP_PYTHON_REPO}/git/refs/heads/${LLAMA_CPP_PYTHON_BRANCH} /tmp/llama_cpp_python_version.json

You will find Dockerfile.minus-github-api file newly created in each package directory if the Dockerfile contains such line, and that's what used for building. The Dockerfile.minus-github-api is temporary (and is listed in .gitignore), so always edit the original Dockerfile when needed. If you want to remove all such temporary files, you can execute the following command.

find . -type f -name *.minus-github-api -delete