Linux Build Basics#
Image Hierarchy#
ubuntu:26.04
└── base (:base)
├── compiler/toolchain (:cross-compiler-amd64)
│ └── sdk (:cross-sdk-<arch>)
│ ├── media (:cross-media-<arch>)
│ │ └── android (:cross-android-<arch>)
│ ├── nvidia (optional) (:toolchain-nvidia)
│ └── amd (optional) (:toolchain-amd)
└── runtime-base (:latest-cross-base-<arch>)
└── package (:latest-cross-package-<arch>)
└── torch/wrapper (:latest-cross-<arch>)
└── manifest (:latest-cross)
Two Build Lanes:
Lane |
Platform |
Purpose |
Tag prefix |
|---|---|---|---|
Cross lane |
|
Compile artifacts for all target arches |
|
Runtime lane |
Target platform |
Package cross artifacts into target-native images |
|
The final release target is the multi-arch manifest ghcr.io/kataglyphis/kataglyphis_beschleuniger:latest-cross, assembled from per-arch wrappers :latest-cross-{amd64,arm64,riscv64}.
See AGENTS.md for the full container architecture documentation.
Build Flow#
The full :latest-cross pipeline:
Cross lane (stages 1-5, all
linux/amd64):base→compiler→sdk→media→android
Runtime lane (stage 6, target platform via QEMU/binfmt for foreign arches):
base→package→torch/wrapper→manifest
The cross-lane stage chain is defined declaratively in linux/scripts/01-core/stage-defs.sh
as CROSS_STAGE_ORDER. Stage orchestration (build, push, pin) is handled by shared functions
in linux/scripts/01-core/cross-stage-build.sh. See docs/linux-cross-builds.md for the
full stage graph API and digest-pinning details.
Prefer the orchestrator for full chains:
bash linux/scripts/build-cross-chain.sh --target-arches amd64,arm64,riscv64
For single-stage rebuilds, use the standalone helpers:
bash linux/scripts/build-cross-stage.sh --stage compiler --push
bash linux/scripts/build-cross-stage.sh --stage sdk --arch arm64 --push
For standalone compiler builds (same as --stage compiler above):
bash linux/scripts/build-cross-compiler.sh --cross-targets amd64,arm64,riscv64
Chain Verification#
# Standalone — fastest way to check staleness:
bash linux/scripts/verify-cross-chain.sh --target-arches amd64,arm64,riscv64
# Via the orchestrator:
bash linux/scripts/build-cross-chain.sh --target-arches amd64,arm64,riscv64 --verify-chain
# Print full stage graph with tag names:
bash linux/scripts/build-cross-chain.sh --target-arches amd64,arm64,riscv64 --describe-chain
See docs/linux-cross-builds.md for the full stale-check reference and --describe-chain output.
Rootless Build Networking (host tuning)#
This host’s rootless BuildKit is tuned for fast build-time downloads. The OCI worker runs with --oci-worker-net=host (via ~/.config/systemd/user/buildkit.service.d/override.conf), so every RUN step (for example the LLVM git fetch in the cross-compiler build) uses host networking instead of the slow rootless bridge/slirp path. With this in place, a plain nerdctl build already uses host networking; you do not need --network host. Docker Hub pulls are mirrored through mirror.gcr.io, but mirrors only speed up FROM ... image pulls, not in-build git/curl downloads. See docs/project-info.md for the exact drop-in files and how to re-apply them. Do not regress these settings.
Build#
# Recommended: cross-lane digest-pinned release
sudo nerdctl run -it --rm ghcr.io/kataglyphis/kataglyphis_beschleuniger:latest-cross
# on Windows you must expose ports one by one
sudo nerdctl run -it --rm -p 8443:8443 ghcr.io/kataglyphis/kataglyphis_beschleuniger:latest-cross
# Legacy QEMU/binfmt image:
# sudo nerdctl run -it --rm ghcr.io/kataglyphis/kataglyphis_beschleuniger:latest
Optional Ubuntu Apt Mirror Workaround#
Add both
--build-arg USE_FAST_UBUNTU_MIRROR=trueand--build-arg FAST_UBUNTU_MIRROR_URL=...when the default Ubuntu archive mirror is slow.Example German mirror override:
--build-arg FAST_UBUNTU_MIRROR_URL=http://de.archive.ubuntu.com/ubuntu/.The helper rewrites archive mirror entries only by default;
security.ubuntu.comstays untouched unless you explicitly opt into rewriting it.Helper scripts expose the same behavior via
--fast-ubuntu-mirrorand--fast-ubuntu-mirror-url.
Generic usage:
sudo nerdctl build \
--build-arg USE_FAST_UBUNTU_MIRROR=true \
--build-arg FAST_UBUNTU_MIRROR_URL=http://de.archive.ubuntu.com/ubuntu/ \
-f <dockerfile> \
.
Supported Dockerfiles:
linux/Dockerfile.baselinux/Dockerfile.toolchainlinux/Dockerfile.sdklinux/Dockerfile.medialinux/Dockerfile.androidlinux/Dockerfile.packagelinux/Dockerfile.nvidialinux/Dockerfile.amdlinux/Dockerfile.torch
Local smoke validation for the shared package+wrapper flow (native mode):
nerdctl build --platform linux/amd64 \
-t local/kataglyphis:latest-cross-wrapper-smoke-amd64 \
-f linux/Dockerfile.package \
--target wrapper-smoke \
--build-arg BASE_IMAGE=ghcr.io/kataglyphis/kataglyphis_beschleuniger:base \
--build-arg ARTIFACT_IMAGE=ghcr.io/kataglyphis/kataglyphis_beschleuniger:android \
--build-arg ARTIFACT_PLATFORM=linux/amd64 \
--build-arg TARGET_ARCH=amd64 \
--build-arg BUILD_MODE=native \
.
Cross-mode variant (validates cross-assembled artifacts):
nerdctl build --platform linux/amd64 \
-t local/kataglyphis:latest-cross-wrapper-smoke-amd64 \
-f linux/Dockerfile.package \
--target wrapper-smoke \
--build-arg BASE_IMAGE=ghcr.io/kataglyphis/kataglyphis_beschleuniger:base \
--build-arg ARTIFACT_IMAGE=ghcr.io/kataglyphis/kataglyphis_beschleuniger:cross-android-amd64 \
--build-arg ARTIFACT_PLATFORM=linux/amd64 \
--build-arg TARGET_ARCH=amd64 \
--build-arg BUILD_MODE=cross \
.
The runtime helpers use the same local-only handoff internally, so --skip-manifest and other non-push runs do not require a registry-visible base/package tag.
build-runtime-manifest.sh --manifest-only(alias--repair) creates/pushes the manifest only, useful for repairing a manifest from existing per-arch wrappers without rebuilding images. They still run the Torch stage on the real target platform so the final image includes/opt/venv.
In cross mode, the media artifact lane makes a best-effort riscv64 app wheelhouse on the amd64 host for the locked torch, torchvision, and opencv-python git-source dependencies used by Kataglyphis-Orchestr-ANT-ion. The final Torch install keeps the upstream uv.lock when present so it can reuse those local wheels instead of re-resolving the same git sources under QEMU. If a reused cross artifact has an empty /opt/wheels, that Torch step keeps the packages that uv sync already resolved instead of trying to install a literal /opt/wheels/*.whl glob.
The package stage must keep /usr/bin/clang pointed at the copied target-native /usr/local/llvm-target/bin/clang on all architectures; do not let it fall back to distro /usr/local/llvm-22 on arm64 or riscv64. On all architectures, /usr/bin/cc points to the source-built /opt/gcc-16.1.0/bin/gcc. On amd64, GCC is built natively. On arm64 and riscv64, GCC is cross-compiled from source (Canadian cross) during the toolchain stage so /opt/gcc-16.1.0/bin/gcc is a target-native binary; Dockerfile.android swaps it in at the end of the Android stage.
On this host, prefer linux/scripts/build-runtime-artifacts.sh and linux/scripts/build-runtime-manifest.sh over ad hoc nerdctl build loops.
When you rebuild a cross SDK artifact from an older cross-compiler-amd64 base, linux/Dockerfile.sdk now forwards the checked-in LLVM_RELEASE into target-clang so /opt/llvm-target is refreshed to the repository pin instead of inheriting a stale base-image environment value.
Those temporary stage contexts now default to ${XDG_CACHE_HOME:-$HOME/.cache}/opencode/runtime-build-contexts and each one is deleted after the next stage finishes using it. Both runtime helpers accept --target-arches, TARGET_ARCHES, and TARGET_ARCH for selecting target architectures.
The main repo-root Linux Dockerfiles now use Dockerfile-specific ignore files too, so routine Linux builds no longer send the large linux/webserver/ tree through the base/toolchain/sdk/media/android/package/torch/wrapper contexts.
When you are feeding locally saved runtime artifacts back into later builds:
Keep
.dockerignoreexcludingout/local-oci,out/local-android-dir,out/linux-sdk,out/linux-runtime, andout/runtime-repair-*so exported OCI layouts and rootfs trees do not get sent back as a later build context.Prefer saved OCI layouts such as
out/local-oci/android/<arch>for foreign-architecture runtime packaging. The plain directory exports underout/local-android-dir/<arch>are much larger, and an earlier OCI-to-directory conversion dropped/usr/local/lib/onnxruntime-cpu.On this host, the verified local runtime path mixes context types: the heavy
runtime_artifactinput comes from anoci-layout://...context, while the intermediateruntime_basehandoff stays a plain rootfs directory because one build still fails when it consumes two named OCI image contexts at once.readlink -fon symlinks insideout/linux-runtime/*/rootfsresolves absolute links against the host root, so use plainreadlinkor validate from inside the built image when checking/usr/bin/cc,/usr/bin/clang,/etc/alternatives/cc, and/etc/alternatives/clang.Local BuildKit and containerd stores can still grow very large during repeated runtime rebuilds, so prune old images and exported artifacts if disk pressure returns.
Not supported / not needed:
linux/webserver/Dockerfileis not wired for this flag.windows/Dockerfiledoes not use apt.
Multi-Arch Build#
QEMU/binfmt: If foreign-architecture builds fail with
exec format error, runsudo nerdctl run --rm --privileged tonistiigi/binfmt --install allin a terminal first. The binfmt registration may need to be reinstalled after a host reboot.
RISC-V64 example#
nerdctl build --platform linux/riscv64 --build-arg GSTREAMER_VERSION=1.29.1 --no-cache \
-t ghcr.io/kataglyphis/kataglyphis_beschleuniger:riscv -f linux/Dockerfile.media \
--cache-to=type=registry,ref=ghcr.io/kataglyphis/kataglyphis_beschleuniger:buildcache,mode=max,oci-mediatypes=true \
--cache-from=type=registry,ref=ghcr.io/kataglyphis/kataglyphis_beschleuniger:buildcache \
.
linux/Dockerfile.torch also exposes a venv-export stage for cases where you only want the built /opt/venv tree:
nerdctl build --platform linux/arm64 \
-f linux/Dockerfile.torch \
--target venv-export \
--output type=local,dest=out/torch-venv/arm64 \
--build-arg BASE_IMAGE=ghcr.io/kataglyphis/kataglyphis_beschleuniger:android \
--build-arg TORCH_APP_MODE=install \
.
For a full hands-off cross build of :latest-cross, prefer the orchestrator linux/scripts/build-cross-chain.sh. It chains base -> compiler -> sdk -> media -> android -> runtime with digest-pinned stage handoff. See docs/linux-cross-builds.md for the full pipeline and AGENTS.md for the stage handoff rules.