Skip to content

Latest commit

 

History

History

Qt6.8forJetson

Cross compilation of Qt6.8.0 Jetson Orin/Nx/Nano with Docker(Base and QML packages) and Remote Debugging with Vscode

In this content, you will find a way to cross-compile Qt 6.8.0 for Jetson Orin/Nx/Nano(ubuntu os) hardware using Docker isolation. This is a complete tutorial that you can learn how to debug the application with vscode.

The primary advantage of Docker is its ability to isolate the build environment. This means you can build Qt without needing a Jetson Development Boards (real hardware) and regardless of your host OS type, as long as you can run Docker (along with QEMU). Additionally, you won’t need to handle dependencies anymore (and I’m not kidding). This approach is easier and less painful.

Watch the video for more details:

Youtube video link

This is previous video, detailed explanation is in here too.

Youtube video link

For remote debugging and follow up click.

I tested this on Ubuntu 22 and 20(as host). Regardless of the version, Qt is successfully cross-compiled and builds a 'Hello World' application (with QML) for Jetson Orin/Nx/Nano.

The steps will show you how to prepare your build environment (in this case, Ubuntu) and run the Docker commands to build Qt 6.8.0. But as I mentioned, you don't need to use Ubuntu; as long as you can run the Docker engine and QEMU, you should achieve the same result on any platform.

If you want to check with virtual machine you can find tutorial Here. Steps are quite same, for this case you need raspberry pi. It is classical way that you can find in this repository. Or If you want more infromation, check old videos about it. If you want to understand theory for cross complation of Qt for rasppberry pi without Docker in detail, you can watch this video which shows how to compile Qt 6.3.0 for raspberry pi(only toolchain is not compiled).

Install Docker

NOTE: If you see error during installation, then search on the internet how to install docker and qemu for your os. During time this steps can be different as you expect.

I have ubuntu 24 (according to ubuntu version steps can vary)

ulas@ulas:~$ lsb_release -a
No LSB modules are available.
Distributor ID:	Ubuntu
Description:	Ubuntu 24.04 LTS
Release:	24.04
Codename:	noble

Lets install dependencies.

# Add Docker's official GPG key:
$ sudo apt-get update
$ sudo apt-get install ca-certificates curl
$ sudo install -m 0755 -d /etc/apt/keyrings
$ sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc
$ sudo chmod a+r /etc/apt/keyrings/docker.asc

Set up stable repository for docker

echo \
  "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \
  $(. /etc/os-release && echo "$VERSION_CODENAME") stable" | \
  sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
$ sudo apt-get update

Install related packages for Docker

sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin

Verify installation with hello-world image

$ sudo docker run hello-world

Lets manage user permission for Docker. Docker uses UDS so permission is needed.

$ sudo usermod -aG docker ${USER}
$ su - ${USER}
$ sudo systemctl enable docker

We also need to install QEMU, with it, it is possible to emulate/run raspbian os like it is on real raspberry pi 4 hardware

$ sudo apt install qemu-system-x86 qemu-kvm libvirt-daemon-system libvirt-clients bridge-utils virt-manager

Enable and Start Libvirt:

$ sudo systemctl enable libvirtd
$ sudo systemctl start libvirtd

Add Your User to the Libvirt and KVM Groups:

$ sudo usermod -aG libvirt $(whoami)
$ sudo usermod -aG kvm $(whoami)

Verify Installation:

$ virsh list --all

You should see an empty list.

Set up QEMU for multi-architecture support

$ docker run --rm --privileged multiarch/qemu-user-static --reset -p yes

Create and use a new Buildx builder instance

$ docker buildx create --use --name mybuilder
$ docker buildx inspect mybuilder --bootstrap

Verify Buildx installation

$ docker buildx ls

Compile Qt 6.8.0 with Docker

When I experimented with this idea, I expected to create a single Dockerfile with different stages, allowing me to switch between them even if they involved different hardware architectures. However, it didn't work as expected, so I ended up creating two separate Dockerfiles.

The tricky part for jetson boards, there can be multiple versions for gcc, jetpack, cuda.. These versions should be handled correctly. You can find a table which shows the paremeters to be passed to dockerfile according to board which Qt is build for.

Compatibility Table

The table below provides details on the compatibility of GCC, Ubuntu, JetPack, CUDA, and Jetson board types:

GCC Version Ubuntu Version JetPack Version CUDA Version BOARD_TYPE Use Case / Notes
7 18.04 r32.x (e.g., r32.7.3) 10-2 t210, t186 Used for JetPack 4.x on Nano (t210) and TX2 (t186). Works with CUDA 10.2 and older.
9 20.04 r35.1, r35.2 11-4 t194, t210 Default for JetPack 5.x with Xavier NX (t194) or Nano (JetPack 5.x experimental).
10 20.04 r35.2 11-6 t194 Optional for advanced builds on JetPack 5.x when GCC 10-specific features are required.
11 22.04 r35.3+ 11-8 t234 Default for JetPack 5.x on Orin (t234) or future boards. Required for CUDA 11.8.

  • note: Old gcc versions can cause conflicts with Qt6.x.x

Notes on JetPack Versions

  • JetPack 4.x: Use r32.x (latest: r32.7.3) for legacy systems based on Ubuntu 18.04 with GCC 7.
    • r32.7.x is the final release for JetPack 4.x and supports Nano (t210) and TX2 (t186).
  • JetPack 5.x: Use r35.x (e.g., r35.1, r35.2, or later) for modern systems based on Ubuntu 20.04+ with GCC 9 or higher.
    • Recommended for Xavier NX (t194) and Orin (t234).
  • CUDA Compatibility:
    • r32.x: CUDA 10.x.
    • r35.x: CUDA 11.x.

First, we will create a ubuntu 20/22 environment and emulate it. Then, we need to copy the relevant headers and libraries for later compilation

Run the command to create ubuntu image. This command is for Nx. But if you change the version numbers according to table then you can build the qt for other boards.

$ docker buildx build --platform linux/arm64 --load -f dockerFileJetson --build-arg UBUNTU_VERSION=20.04 --build-arg CUDA_VERSION=11-4 --build-arg UVUNTU_VERSION_SHORT=2004 --build-arg JETPACK_VERSION=r35.2 --build-arg BOARD_TYPE=t194 -t jetsonimage .

When it finishes, you will find a file named 'jetsonSysroot.tar.gz' in the '/' directory within the image. Let's copy it to the same location where the Dockerfile exists.

To copy the file, you need to create a temporary container using the 'create' command. You can delete this temporary container later if you wish

$ docker create --name temp-arm jetsonimage
$ docker cp temp-arm:/jetsonSysroot.tar.gz ./jetsonSysroot.tar.gz

This jetsonSysroot.tar.gz file will be copied by the another image that is why location of the tar file is important. You do not need to extract it. Do not touch it.

Now it is time to create ubuntu 22 image and compile the Qt 6.8.0. In one of the previous commands you used DockerFileNx, this file is written for Jetson boards, now we are going to use only Dockerfile which is default name that means we do not need to specify path or name explicitly. But if you want you can change the name, you already now how you can pass the file name (with -f)

$ docker build --build-arg UBUNTU_VERSION=20.04 --build-arg GCC_VERSION=9 -t qtcrossjet .

As you see there is no buildx in this command because buildx uses qemu and we do not need qemu for x86 ubuntu. After some time, (I tested with 16GB RAM and it took around couple of hours) you see that image will be created without an error. After this, you can find HelloQt6 binary which is ready to run on Jetson board, in the /project directory in the image. So lets copy it. As we did before, you need to create temporary container to copy it.

$ docker create --name tmpbuild qtcrossjet
$ docker cp tmpbuild:/project/HelloQt6 ./HelloQt6

As you see, example application is compiled for arm.

ulas@ulas:~/QtJetsonNx$ file HelloQt6 
HelloQt6: ELF 64-bit LSB executable, ARM aarch64, version 1 (SYSV), dynamically linked, interpreter /lib/ld-linux-aarch64.so.1, for GNU/Linux 3.7.0, with debug_info, not stripped

To test the hello world, you need to copy and send the compiled qt binaries in the image.

$ docker cp tmpbuild:/qt-jetson-binaries.tar.gz ./qt-jetson-binaries.tar.gz
$ scp qt-jetson-binaries.tar.gz ulas@192.168.16.20:/home/ulas/
$ ssh ulas@192.168.16.25
$ ulas@jetson:~ sudo mkdir /usr/local/qt6
$ ulas@jetson:~ sudo tar -xvf qt-jetson-binaries.tar.gz -C /usr/local/qt6
$ ulas@jetson:~ export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/usr/local/qt6/lib/

Extract it under /usr/local or wherever you want and do not forget to add the path to LD_LIBRARY_PATH in case of path is not in the list.

ulas@Jetson:~ $ ./HelloQt6
Hello world

Debugging of compilation

Nothing is free! Okay now we find a nice way to compile or build Qt applications but there is a tradeoff. Debugging is really hard. So If you want to change Dockerfile then first you sould build or test the steps on VM to be sure. If you know what you are doing then do not worry.

Each RUN commands output are printed in Build.log file that you can find in the build directory of image.

docker cp tmpbuild:/build.log ./build.log

Cross Development and Remote Debugging of Application with vscode ( this is tested with rasp but you can adjust it accordingly)

Detailed information:

Youtube video link

Now, you can build your application by simply adding your files to the project directory and running the command:

$ docker build -t qtcrossjet .

If you do not modify the Dockerfile, the above command will only compile the code. However, even if you don't alter the Dockerfile, if there is any update to Ubuntu, Docker will rebuild the image starting from the Ubuntu layer. This means it will take more time. To compile the Qt application you wish to develop, this is not necessary. There is another Dockerfile, Dockerfile.app, which allows you to compile only the application. If you examine the contents of this file, you will find:

FROM qtcrossjet:latest

This indicates that if you build an image with Dockerfile.app, it will utilize the qtcrossjet image. There's no need to run qtcrossjet again.

If you run:

$ docker build -f Dockerfile.app -t final-app .

With the final-app image, you can create a container solely for compilation purposes. Docker caches the previous commands, so when you run this command, it will not start from scratch but only execute the latest command where you wish to compile your application. The compilation process will begin in the image, then, as before, create a temporary container and copy your binary:

$ docker create --name tmpapp final-app
$ docker cp tmpapp:/projectPath/HelloQt6 ./HelloQt6

If you do not want to use the cache, or if you want to start building the same image anew, use:

$ docker build -t qtcrossjet . --no-cache

However, if you prefer not to follow these steps, I have shared the tar files that I compiled for the Raspberry Pi, along with the related sysroot and toolchain. You can download them. In this case, you will need to have the correct dependencies. It's your choice.

I assume you have vscode already, we need some dependencies on the host for remote debugging with vscode.

$ sudo apt-get install sshpass gdb-multiarch

sshpass is needed to start gdbserver on the target remotely, and gdb-multiarch is the debugger itself for the cross-compiled binary.

In VS Code, you will need the "C/C++ IntelliSense, debugging, and code browsing" extension.

Qml Remote Debugging with vscode

On the target, install gdbserver.

$ sudo apt-get install gdbserver

To debug the cross-compiled application, which was compiled in the container, we can use VS Code. The steps are simple:

  1. Compile the application using Dockerfile.app.
  2. Copy the binary of the application to the target.
  3. Run gdbserver on the target before debug process.
  4. Connect to the server from the host using VS Code, then start debugging.

There are multiple ways to implement these steps. I have created a bash script called helperTasks.sh. This script contains commands to compile the application, send the binary to the target, and start gdbserver. The script acts as an intermediary between VS Code and the user. When you want to call one of these functionalities, VS Code may access them through tasks.json. As mentioned, there are many ways to do this; you can directly update tasks.json as well. Tasks.json is located under .vscode in the repository. If you start VS Code directly in the root of the repository, it will automatically detect related files, and everything will be ready to go.

For debugging itself, VS Code needs a launch.json configuration file. In this file, you need to specify the path of the application and other related options. However, be careful; the paths for the binary should be on the host, so a copy of the binary must also exist on the host.

One trick to note is the importance of the checkout path. When you compile the application in the container (with project and projectQml as ready-to-use examples), the binary contains information about the absolute path of the project files, which were copied from the host to the container. It has the path in the container, and when you copy it to the host, it doesn't recognize that the path has changed. Therefore, you need to copy the binary to the exact same location on the host, or you need to change the path in the container. That's why I created a variable for the path in both Dockerfile.app and helperTasks.sh. Please check and update them accordingly.

Qml Remote Debugging with vscode

For more detalied information please watch the video about debugging.

Enjoy.

Configuration parameters

For reference the qt is compiled with below parameters in this example

-- Configure summary:

Building for: linux-g++ (arm64, CPU features: cx16 neon)
Compiler: gcc 9.4.0
Build options:
  Mode ................................... release
  Optimize release build for size ........ no
  Fully optimize release builds (-O3) .... no
  Building shared libraries .............. yes
  Using ccache ........................... no
  Unity Build ............................ no
  Using new DTAGS ........................ yes
  Relocatable ............................ yes
  Using precompiled headers .............. yes
  Using Link Time Optimization (LTCG) .... no
  Using Intel CET ........................ no
  Target compiler supports:
    ARM Extensions ....................... NEON
  Sanitizers:
    Addresses ............................ no
    Threads .............................. no
    Memory ............................... no
    Fuzzer (instrumentation only) ........ no
    Undefined ............................ no
  Build parts ............................ libs
  Install examples sources ............... no
Qt modules and options:
  Qt Concurrent .......................... yes
  Qt D-Bus ............................... yes
  Qt D-Bus directly linked to libdbus .... no
  Qt Gui ................................. yes
  Qt Network ............................. yes
  Qt PrintSupport ........................ yes
  Qt Sql ................................. yes
  Qt Testlib ............................. yes
  Qt Widgets ............................. yes
  Qt Xml ................................. yes
Support enabled for:
  Using pkg-config ....................... yes
  Using vcpkg ............................ no
  udev ................................... yes
  OpenSSL ................................ no
    Qt directly linked to OpenSSL ........ no
  OpenSSL 1.1 ............................ no
  OpenSSL 3.0 ............................ no
  Using system zlib ...................... yes
  Zstandard support ...................... no
  Thread support ......................... yes
Common build options:
  Linker can resolve circular dependencies  yes
Qt Core:
  backtrace .............................. yes
  DoubleConversion ....................... yes
    Using system DoubleConversion ........ no
  CLONE_PIDFD support in forkfd .......... yes
  GLib ................................... no
  ICU .................................... no
  Using system libb2 ..................... no
  Built-in copy of the MIME database ..... yes
  Application permissions ................ yes
  Defaulting legacy IPC to POSIX ......... no
  Tracing backend ........................ <none>
  OpenSSL based cryptographic hash ....... no
  Logging backends:
    journald ............................. no
    syslog ............................... no
    slog2 ................................ no
  PCRE2 .................................. yes
    Using system PCRE2 ................... no
Qt Sql:
  SQL item models ........................ yes
Qt Network:
  getifaddrs() ........................... yes
  IPv6 ifname ............................ yes
  libproxy ............................... no
  Linux AF_NETLINK ....................... yes
  DTLS ................................... no
  OCSP-stapling .......................... no
  SCTP ................................... no
  Use system proxies ..................... yes
  GSSAPI ................................. no
  Brotli Decompression Support ........... no
  qIsEffectiveTLD() ...................... yes
    Built-in publicsuffix database ....... yes
    System publicsuffix database ......... yes
Core tools:
  Android deployment tool ................ no
  macOS deployment tool .................. no
  Windows deployment tool ................ no
  qmake .................................. yes
Qt Gui:
  Accessibility .......................... yes
  FreeType ............................... yes
    Using system FreeType ................ yes
  HarfBuzz ............................... yes
    Using system HarfBuzz ................ no
  Fontconfig ............................. yes
  Image formats:
    GIF .................................. yes
    ICO .................................. yes
    JPEG ................................. yes
      Using system libjpeg ............... no
    PNG .................................. yes
      Using system libpng ................ yes
  Text formats:
    HtmlParser ........................... yes
    CssParser ............................ yes
    OdfWriter ............................ yes
    MarkdownReader ....................... yes
      Using system libmd4c ............... no
    MarkdownWriter ....................... yes
  EGL .................................... yes
  OpenVG ................................. no
  OpenGL:
    Desktop OpenGL ....................... no
    OpenGL ES 2.0 ........................ yes
    OpenGL ES 3.0 ........................ yes
    OpenGL ES 3.1 ........................ yes
    OpenGL ES 3.2 ........................ yes
  Vulkan ................................. no
  Session Management ..................... yes
Features used by QPA backends:
  evdev .................................. yes
  libinput ............................... no
  HiRes wheel support in libinput ........ no
  INTEGRITY HID .......................... no
  mtdev .................................. no
  tslib .................................. no
  xkbcommon .............................. yes
  X11 specific:
    xlib ................................. yes
    XCB Xlib ............................. yes
    EGL on X11 ........................... yes
    xkbcommon-x11 ........................ yes
    xcb-sm ............................... no
QPA backends:
  DirectFB ............................... no
  EGLFS .................................. yes
  EGLFS details:
    EGLFS OpenWFD ........................ no
    EGLFS i.Mx6 .......................... no
    EGLFS i.Mx6 Wayland .................. no
    EGLFS RCAR ........................... no
    EGLFS EGLDevice ...................... yes
    EGLFS GBM ............................ yes
    EGLFS VSP2 ........................... no
    EGLFS Mali ........................... no
    EGLFS Raspberry Pi ................... no
    EGLFS X11 ............................ yes
  LinuxFB ................................ yes
  VNC .................................... yes
  VK_KHR_display ......................... no
  QNX:
    lgmon ................................ no
    IMF .................................. no
  XCB:
    Using system-provided xcb-xinput ..... no
    GL integrations:
      GLX Plugin ......................... no
        XCB GLX .......................... no
      EGL-X11 Plugin ..................... yes
  Windows:
    Direct 2D ............................ no
    Direct 2D 1.1 ........................ no
    DirectWrite .......................... no
    DirectWrite 3 ........................ no
Qt Widgets:
  GTK+ ................................... no
  Styles ................................. Fusion Windows
Qt Testlib:
  Tester for item models ................. yes
  Batch tests ............................ no
Qt PrintSupport:
  CUPS ................................... no
Qt Sql Drivers:
  DB2 (IBM) .............................. no
  InterBase .............................. no
  MySql .................................. no
  OCI (Oracle) ........................... no
  ODBC ................................... no
  PostgreSQL ............................. no
  SQLite ................................. yes
    Using system provided SQLite ......... no
  Mimer .................................. no
 

Note: Disabling X11 Accessibility Bridge: D-Bus or AT-SPI is missing.
Note: Due to CMAKE_STAGING_PREFIX usage and an unfixed CMake bug,
      to ensure correct build time rpaths, directory-level install
      rules like ninja src/gui/install will not work.
      Check QTBUG-102592 for further details.

-- 

Qt is now configured for building. Just run 'cmake --build . --parallel'

Once everything is built, you must run 'cmake --install .'
Qt will be installed into '/usr/local/qt6'