|
|
@@ -12,14 +12,27 @@
|
|
|
|
# it would need a glibc (Debian) base and produces a much larger image. See
|
|
|
|
# it would need a glibc (Debian) base and produces a much larger image. See
|
|
|
|
# README for details if you need it.
|
|
|
|
# README for details if you need it.
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# The Go builder cross-compiles, so it always runs NATIVELY on the build host
|
|
|
|
# Both the Go (Tailscale) stage and the C (busybox) stage cross-compile: they
|
|
|
|
# ($BUILDPLATFORM) for speed; only the busybox stage and the final image run on
|
|
|
|
# always run NATIVELY on the build host ($BUILDPLATFORM) and produce binaries
|
|
|
|
# the target platform.
|
|
|
|
# for $TARGETPLATFORM. This eliminates QEMU emulation entirely from the build,
|
|
|
|
|
|
|
|
# which is the main source of slowness in multi-arch builds. Only the final
|
|
|
|
|
|
|
|
# scratch stage pulls in the target-arch-specific layers (CA certs, busybox
|
|
|
|
|
|
|
|
# rootfs) which are just file copies with no emulated execution.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# Cross-compilation for C (busybox) is provided by tonistiigi/xx, which
|
|
|
|
|
|
|
|
# configures clang+lld as a cross-compiler and installs musl headers for the
|
|
|
|
|
|
|
|
# target arch via xx-apk.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# =============================================================================
|
|
|
|
|
|
|
|
# xx: Dockerfile cross-compilation helpers (provides xx-clang, xx-apk, etc.)
|
|
|
|
|
|
|
|
# =============================================================================
|
|
|
|
|
|
|
|
# renovate: datasource=docker depName=tonistiigi/xx versioning=docker
|
|
|
|
|
|
|
|
FROM --platform=$BUILDPLATFORM tonistiigi/xx:1.9.0@sha256:c64defb9ed5a91eacb37f96ccc3d4cd72521c4bd18d5442905b95e2226b0e707 AS xx
|
|
|
|
|
|
|
|
|
|
|
|
# =============================================================================
|
|
|
|
# =============================================================================
|
|
|
|
# Stage 1: Build Tailscale combined binary (cross-compiled, runs natively)
|
|
|
|
# Stage 1: Build Tailscale combined binary (cross-compiled, runs natively)
|
|
|
|
# =============================================================================
|
|
|
|
# =============================================================================
|
|
|
|
FROM --platform=$BUILDPLATFORM golang:1.26.4-alpine@sha256:7a3e50096189ad57c9f9f865e7e4aa8585ed1585248513dc5cda498e2f41812c AS builder
|
|
|
|
FROM --platform=$BUILDPLATFORM golang:1.26.4-alpine@sha256:f1ddd9fe14fffc091dd98cb4bfa999f32c5fc77d2f2305ea9f0e2595c5437c14 AS builder
|
|
|
|
|
|
|
|
|
|
|
|
# renovate: datasource=github-releases depName=tailscale packageName=tailscale/tailscale versioning=semver
|
|
|
|
# renovate: datasource=github-releases depName=tailscale packageName=tailscale/tailscale versioning=semver
|
|
|
|
ARG TAILSCALE_VERSION=v1.98.5
|
|
|
|
ARG TAILSCALE_VERSION=v1.98.5
|
|
|
@@ -57,49 +70,6 @@ WORKDIR /src/tailscale
|
|
|
|
# disables the filter at runtime for debugging — no rebuild needed.
|
|
|
|
# disables the filter at runtime for debugging — no rebuild needed.
|
|
|
|
COPY patches/stderr_verbosity_filter.go cmd/tailscaled/
|
|
|
|
COPY patches/stderr_verbosity_filter.go cmd/tailscaled/
|
|
|
|
|
|
|
|
|
|
|
|
# Patch net/tstun/wrap.go: fix panic("unreachable") in invertGSOChecksum for
|
|
|
|
|
|
|
|
# ts_omit_netstack builds.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# invertGSOChecksum is a gVisor/GSO helper that inverts a transport-layer
|
|
|
|
|
|
|
|
# checksum before/after SNAT when gVisor hands us a segment with a partial
|
|
|
|
|
|
|
|
# checksum (NeedsCsum=true). It is only meaningful when netstack (gVisor) is
|
|
|
|
|
|
|
|
# compiled in (HasNetstack=true).
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# The function correctly guards its body with:
|
|
|
|
|
|
|
|
# if !buildfeatures.HasNetstack { panic("unreachable") }
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# When built with ts_omit_netstack, HasNetstack is a const false, so that guard
|
|
|
|
|
|
|
|
# evaluates to `if true { panic(...) }` — the function always panics.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# The problem: invertGSOChecksum is called unconditionally from injectedRead()
|
|
|
|
|
|
|
|
# (twice, around pc.snat()), even for the res.data path where res.packet==nil
|
|
|
|
|
|
|
|
# and gso is a zero-value netstack_GSO (NeedsCsum=false). The HasNetstack
|
|
|
|
|
|
|
|
# guard in the res.packet branch does NOT protect these calls.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# As a result, any code path that injects an outbound packet via InjectOutbound()
|
|
|
|
|
|
|
|
# — which happens when enabling exit-node use (Tailscale sends TSMP messages
|
|
|
|
|
|
|
|
# and synthesizes packets through the TUN injection path) — hits injectedRead
|
|
|
|
|
|
|
|
# with res.data!=nil, calls invertGSOChecksum, and crashes with:
|
|
|
|
|
|
|
|
# panic: unreachable
|
|
|
|
|
|
|
|
# tailscale.com/net/tstun.invertGSOChecksum(...)
|
|
|
|
|
|
|
|
# tailscale.com/net/tstun.(*Wrapper).injectedRead(...) wrap.go:1077
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# Fix: replace the `panic("unreachable")` with a `return` in invertGSOChecksum.
|
|
|
|
|
|
|
|
# When HasNetstack=false (ts_omit_netstack), a zero-value netstack_GSO always
|
|
|
|
|
|
|
|
# has NeedsCsum=false, so the function is correctly a no-op anyway. This matches
|
|
|
|
|
|
|
|
# what the function would do if the rest of its body ran: NeedsCsum=false → return.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# The sed expression targets the function precisely: it matches the three-line
|
|
|
|
|
|
|
|
# sequence that opens invertGSOChecksum's HasNetstack guard, and replaces only
|
|
|
|
|
|
|
|
# the panic line with return. The pattern is stable across minor reformats
|
|
|
|
|
|
|
|
# because it anchors on the literal function comment and the specific panic string.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# See tailscale/tailscale issue for context (no upstream fix as of v1.98.5):
|
|
|
|
|
|
|
|
# panic happens when using exit-node via a ts_omit_netstack build.
|
|
|
|
|
|
|
|
RUN sed -i \
|
|
|
|
|
|
|
|
-e '/func invertGSOChecksum/,/^}/ s/\t\tpanic("unreachable")/\t\treturn/' \
|
|
|
|
|
|
|
|
net/tstun/wrap.go
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Build a minimal combined binary (tailscale CLI + tailscaled daemon in one file).
|
|
|
|
# Build a minimal combined binary (tailscale CLI + tailscaled daemon in one file).
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# Tag strategy — ALLOWLIST, not blocklist:
|
|
|
|
# Tag strategy — ALLOWLIST, not blocklist:
|
|
|
@@ -135,6 +105,34 @@ RUN sed -i \
|
|
|
|
# waiting for completion") WITHOUT printing the auth URL
|
|
|
|
# waiting for completion") WITHOUT printing the auth URL
|
|
|
|
# or confirming success. Including it makes interactive
|
|
|
|
# or confirming success. Including it makes interactive
|
|
|
|
# 'up' behave normally (blocks, prints login URL).
|
|
|
|
# 'up' behave normally (blocks, prints login URL).
|
|
|
|
|
|
|
|
# netstack — gVisor userspace network stack. Counter-intuitively
|
|
|
|
|
|
|
|
# REQUIRED even though the router uses a real kernel TUN
|
|
|
|
|
|
|
|
# (NOT --tun=userspace-networking). In v1.98.5 the
|
|
|
|
|
|
|
|
# 100.100.100.100:53 MagicDNS listener is served ONLY by
|
|
|
|
|
|
|
|
# netstack's handleLocalPackets, installed via
|
|
|
|
|
|
|
|
# PreFilterPacketOutboundToWireGuardNetstackIntercept.
|
|
|
|
|
|
|
|
# The non-netstack "engine" interceptor that the wrap.go
|
|
|
|
|
|
|
|
# comments claim handles quad-100 "if netstack is not
|
|
|
|
|
|
|
|
# installed" does NOT actually do so on Linux (its body
|
|
|
|
|
|
|
|
# only reflects loopback on darwin/ios/plan9, else
|
|
|
|
|
|
|
|
# Accept). So with ts_omit_netstack, NOTHING absorbs
|
|
|
|
|
|
|
|
# packets to 100.100.100.100: queries fall through to
|
|
|
|
|
|
|
|
# WireGuard, no peer owns that IP, and even tailnet-name
|
|
|
|
|
|
|
|
# resolution (and 'ping host.tailnet.ts.net') times out.
|
|
|
|
|
|
|
|
# The 'dns' tag links the resolver but nothing routes
|
|
|
|
|
|
|
|
# packets to it without netstack — the two tags are
|
|
|
|
|
|
|
|
# independent (dns has no Dep on netstack). Omitting
|
|
|
|
|
|
|
|
# netstack ALSO triggered a panic("unreachable") in
|
|
|
|
|
|
|
|
# net/tstun.invertGSOChecksum on the exit-node inject
|
|
|
|
|
|
|
|
# path (HasNetstack=const false made the guard always
|
|
|
|
|
|
|
|
# panic); enabling netstack makes that guard dead code,
|
|
|
|
|
|
|
|
# fixing the crash as a side effect. Cost (arm64, vs a
|
|
|
|
|
|
|
|
# netstack-omitted build): ~+0.5 MB extracted on flash
|
|
|
|
|
|
|
|
# and ~+2.3 MB resident RAM after UPX decompression —
|
|
|
|
|
|
|
|
# measured, acceptable for a 16 MB-flash router.
|
|
|
|
|
|
|
|
# gro — Generic Receive Offload (perf). Depends on netstack;
|
|
|
|
|
|
|
|
# pulled in with it. Small, and improves throughput on
|
|
|
|
|
|
|
|
# the netstack DNS/inject path.
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# Everything else remains omitted, including (rationale):
|
|
|
|
# Everything else remains omitted, including (rationale):
|
|
|
|
# clientupdate — DELIBERATELY removed. The built-in updater would download
|
|
|
|
# clientupdate — DELIBERATELY removed. The built-in updater would download
|
|
|
@@ -159,9 +157,11 @@ RUN sed -i \
|
|
|
|
# which is exactly the flash wear we want to avoid.
|
|
|
|
# which is exactly the flash wear we want to avoid.
|
|
|
|
# logtail — no persistent log writes to flash; also pass
|
|
|
|
# logtail — no persistent log writes to flash; also pass
|
|
|
|
# --no-logs-no-support at runtime
|
|
|
|
# --no-logs-no-support at runtime
|
|
|
|
# netstack+gro — userspace networking; router uses kernel TUN
|
|
|
|
|
|
|
|
# ssh — not needed; access via MikroTik SSH + tailscale CLI
|
|
|
|
# ssh — not needed; access via MikroTik SSH + tailscale CLI
|
|
|
|
# all GUI/desktop/cloud/k8s features — irrelevant for a headless router
|
|
|
|
# all GUI/desktop/cloud/k8s features — irrelevant for a headless router
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# NOTE: netstack/gro are NOT in this omit list — see the opted-in section above
|
|
|
|
|
|
|
|
# for why MagicDNS quad-100 serving structurally requires them in v1.98.5.
|
|
|
|
|
|
|
|
|
|
|
|
RUN mkdir -p /out && \
|
|
|
|
RUN mkdir -p /out && \
|
|
|
|
ALL_OMIT=$(GOOS= GOARCH= go run ./cmd/featuretags --min --add=osrouter) && \
|
|
|
|
ALL_OMIT=$(GOOS= GOARCH= go run ./cmd/featuretags --min --add=osrouter) && \
|
|
|
@@ -178,6 +178,8 @@ RUN mkdir -p /out && \
|
|
|
|
-e 's/ts_omit_iptables,\{0,1\}//g' \
|
|
|
|
-e 's/ts_omit_iptables,\{0,1\}//g' \
|
|
|
|
-e 's/ts_omit_unixsocketidentity,\{0,1\}//g' \
|
|
|
|
-e 's/ts_omit_unixsocketidentity,\{0,1\}//g' \
|
|
|
|
-e 's/ts_omit_ipnbus,\{0,1\}//g' \
|
|
|
|
-e 's/ts_omit_ipnbus,\{0,1\}//g' \
|
|
|
|
|
|
|
|
-e 's/ts_omit_netstack,\{0,1\}//g' \
|
|
|
|
|
|
|
|
-e 's/ts_omit_gro,\{0,1\}//g' \
|
|
|
|
-e 's/,$//' \
|
|
|
|
-e 's/,$//' \
|
|
|
|
) && \
|
|
|
|
) && \
|
|
|
|
echo "Build tags: ${TAGS}" && \
|
|
|
|
echo "Build tags: ${TAGS}" && \
|
|
|
@@ -236,7 +238,7 @@ RUN printf '%s\n' \
|
|
|
|
chmod +x /out/usrlocalbin/entrypoint.sh
|
|
|
|
chmod +x /out/usrlocalbin/entrypoint.sh
|
|
|
|
|
|
|
|
|
|
|
|
# =============================================================================
|
|
|
|
# =============================================================================
|
|
|
|
# Stage 2: Custom minimal busybox
|
|
|
|
# Stage 2: Custom minimal busybox (cross-compiled, runs natively on build host)
|
|
|
|
# =============================================================================
|
|
|
|
# =============================================================================
|
|
|
|
# The official busybox:musl image ships all ~404 applets at ~1.24 MB. For a
|
|
|
|
# The official busybox:musl image ships all ~404 applets at ~1.24 MB. For a
|
|
|
|
# debug shell on a flash-constrained router we only need ~100 applets, so we
|
|
|
|
# debug shell on a flash-constrained router we only need ~100 applets, so we
|
|
|
@@ -253,15 +255,56 @@ RUN printf '%s\n' \
|
|
|
|
# acceptable for an occasional debug shell. RouterOS stores the EXTRACTED
|
|
|
|
# acceptable for an occasional debug shell. RouterOS stores the EXTRACTED
|
|
|
|
# rootfs on disk (overlayfs), so the ~190 kB UPX saving is real on-disk space.
|
|
|
|
# rootfs on disk (overlayfs), so the ~190 kB UPX saving is real on-disk space.
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# This stage runs on the TARGET platform (no --platform override): gcc then
|
|
|
|
# This stage runs NATIVELY on the build host (--platform=$BUILDPLATFORM) and
|
|
|
|
# produces native target-arch binaries directly. Under buildx this is
|
|
|
|
# cross-compiles busybox for the target architecture using clang+lld via the
|
|
|
|
# transparently emulated via binfmt/QEMU for non-native targets.
|
|
|
|
# tonistiigi/xx helpers. This eliminates QEMU emulation from the busybox build,
|
|
|
|
FROM alpine:3.24.0@sha256:a2d49ea686c2adfe3c992e47dc3b5e7fa6e6b5055609400dc2acaeb241c829f4 AS busybox
|
|
|
|
# which was the main source of slowness for arm64/arm/v7 targets.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# Cross-compilation setup:
|
|
|
|
|
|
|
|
# - xx-apk installs musl-dev and linux-headers for the TARGET arch under
|
|
|
|
|
|
|
|
# /<triple> (a secondary sysroot), while clang/lld/upx/make stay native.
|
|
|
|
|
|
|
|
# - xx-clang --setup-target-triple creates <triple>-clang / <triple>-cc
|
|
|
|
|
|
|
|
# aliases in PATH that busybox's Makefile picks up via CROSS_COMPILE.
|
|
|
|
|
|
|
|
# - Busybox make receives:
|
|
|
|
|
|
|
|
# CROSS_COMPILE=<triple>- → picks up <triple>-clang (from xx aliases)
|
|
|
|
|
|
|
|
# CC=clang → use clang (aliased target via CROSS_COMPILE)
|
|
|
|
|
|
|
|
# HOSTCC=gcc → compile host helper tools with native gcc
|
|
|
|
|
|
|
|
# - upx (native x86_64 binary) can compress target-arch binaries since UPX
|
|
|
|
|
|
|
|
# operates on the ELF file format regardless of the target ISA.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# Applet symlink probing: for native-arch builds the probe runs directly;
|
|
|
|
|
|
|
|
# for cross-compiled binaries we use QEMU user-mode emulation (from binfmt)
|
|
|
|
|
|
|
|
# only for this one lightweight probe step (busybox --help per applet), not
|
|
|
|
|
|
|
|
# for the compile itself. The probe can alternatively be skipped by using
|
|
|
|
|
|
|
|
# a pre-enumerated applet list, but the current approach is simpler.
|
|
|
|
|
|
|
|
FROM --platform=$BUILDPLATFORM alpine:3.24.0@sha256:a2d49ea686c2adfe3c992e47dc3b5e7fa6e6b5055609400dc2acaeb241c829f4 AS busybox
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Copy xx cross-compilation helpers (xx-clang, xx-apk, xx-info, etc.)
|
|
|
|
|
|
|
|
COPY --from=xx / /
|
|
|
|
|
|
|
|
|
|
|
|
# renovate: datasource=docker depName=busybox versioning=docker
|
|
|
|
# renovate: datasource=docker depName=busybox versioning=docker
|
|
|
|
ARG BUSYBOX_VERSION=1.38.0
|
|
|
|
ARG BUSYBOX_VERSION=1.38.0
|
|
|
|
|
|
|
|
|
|
|
|
RUN apk add --no-cache build-base linux-headers wget bzip2 perl upx
|
|
|
|
# Target platform ARGs (provided automatically by buildx).
|
|
|
|
|
|
|
|
ARG TARGETPLATFORM
|
|
|
|
|
|
|
|
ARG TARGETARCH
|
|
|
|
|
|
|
|
ARG TARGETVARIANT
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Native build tools (clang/lld for cross-compiling; gcc/make/upx run natively).
|
|
|
|
|
|
|
|
# xx-apk installs the target-arch sysroot: musl-dev (C library headers + CRT),
|
|
|
|
|
|
|
|
# gcc (provides crtbeginS.o/crtendS.o and libgcc needed by clang on Alpine),
|
|
|
|
|
|
|
|
# and linux-headers (required by busybox for <linux/*.h> / <net/*.h>).
|
|
|
|
|
|
|
|
RUN apk add --no-cache \
|
|
|
|
|
|
|
|
clang \
|
|
|
|
|
|
|
|
lld \
|
|
|
|
|
|
|
|
llvm \
|
|
|
|
|
|
|
|
gcc \
|
|
|
|
|
|
|
|
make \
|
|
|
|
|
|
|
|
wget \
|
|
|
|
|
|
|
|
bzip2 \
|
|
|
|
|
|
|
|
perl \
|
|
|
|
|
|
|
|
upx && \
|
|
|
|
|
|
|
|
xx-apk add --no-cache musl-dev gcc linux-headers
|
|
|
|
|
|
|
|
|
|
|
|
RUN wget -q https://busybox.net/downloads/busybox-${BUSYBOX_VERSION}.tar.bz2 \
|
|
|
|
RUN wget -q https://busybox.net/downloads/busybox-${BUSYBOX_VERSION}.tar.bz2 \
|
|
|
|
&& tar xf busybox-${BUSYBOX_VERSION}.tar.bz2
|
|
|
|
&& tar xf busybox-${BUSYBOX_VERSION}.tar.bz2
|
|
|
@@ -269,7 +312,34 @@ WORKDIR /busybox-${BUSYBOX_VERSION}
|
|
|
|
|
|
|
|
|
|
|
|
# allnoconfig = every feature OFF; then enable only the curated applet set.
|
|
|
|
# allnoconfig = every feature OFF; then enable only the curated applet set.
|
|
|
|
COPY busybox-applets.config /tmp/applets.config
|
|
|
|
COPY busybox-applets.config /tmp/applets.config
|
|
|
|
RUN make allnoconfig && \
|
|
|
|
# Set up xx cross-compiler aliases (<triple>-clang, <triple>-cc, etc.) and
|
|
|
|
|
|
|
|
# build busybox.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# Key make variables:
|
|
|
|
|
|
|
|
# ARCH — busybox ARCH; must match the cross-target, not the build
|
|
|
|
|
|
|
|
# host. busybox's Makefile would otherwise read SUBARCH from
|
|
|
|
|
|
|
|
# `uname -m` (the BUILD host's arch) which is wrong when
|
|
|
|
|
|
|
|
# cross-compiling. We map TARGETARCH to busybox's arch name.
|
|
|
|
|
|
|
|
# busybox uses -include arch/$(ARCH)/Makefile; missing arch
|
|
|
|
|
|
|
|
# dirs are silently ignored, so any value is safe.
|
|
|
|
|
|
|
|
# CC — busybox defaults to $(CROSS_COMPILE)gcc. We override CC to
|
|
|
|
|
|
|
|
# the full <triple>-clang path so it resolves to the xx alias
|
|
|
|
|
|
|
|
# (which sets --target and --sysroot for the cross-compiler).
|
|
|
|
|
|
|
|
# Setting CC= avoids needing a <triple>-gcc symlink.
|
|
|
|
|
|
|
|
# HOSTCC — native compiler for host-side build tools (scripts/kconfig,
|
|
|
|
|
|
|
|
# gen_build_files, etc.); must NOT be the cross-compiler.
|
|
|
|
|
|
|
|
# SKIP_STRIP — defer stripping to after symlink probing (we strip below
|
|
|
|
|
|
|
|
# with llvm-strip, which handles any target ELF arch).
|
|
|
|
|
|
|
|
RUN xx-clang --setup-target-triple && \
|
|
|
|
|
|
|
|
CROSS=$(xx-info triple) && \
|
|
|
|
|
|
|
|
# Map TARGETARCH to the busybox ARCH value.
|
|
|
|
|
|
|
|
case "${TARGETARCH}" in \
|
|
|
|
|
|
|
|
amd64) BUSYBOX_ARCH=x86_64 ;; \
|
|
|
|
|
|
|
|
arm64) BUSYBOX_ARCH=aarch64 ;; \
|
|
|
|
|
|
|
|
arm) BUSYBOX_ARCH=arm ;; \
|
|
|
|
|
|
|
|
*) BUSYBOX_ARCH=${TARGETARCH} ;; \
|
|
|
|
|
|
|
|
esac && \
|
|
|
|
|
|
|
|
make allnoconfig ARCH="${BUSYBOX_ARCH}" && \
|
|
|
|
while read -r sym; do \
|
|
|
|
while read -r sym; do \
|
|
|
|
case "$sym" in ''|\#*) continue ;; esac; \
|
|
|
|
case "$sym" in ''|\#*) continue ;; esac; \
|
|
|
|
if grep -q "^# CONFIG_${sym} is not set" .config; then \
|
|
|
|
if grep -q "^# CONFIG_${sym} is not set" .config; then \
|
|
|
@@ -278,9 +348,15 @@ RUN make allnoconfig && \
|
|
|
|
echo "CONFIG_${sym}=y" >> .config; \
|
|
|
|
echo "CONFIG_${sym}=y" >> .config; \
|
|
|
|
fi; \
|
|
|
|
fi; \
|
|
|
|
done < /tmp/applets.config && \
|
|
|
|
done < /tmp/applets.config && \
|
|
|
|
yes "" | make oldconfig >/dev/null 2>&1 && \
|
|
|
|
yes "" | make oldconfig ARCH="${BUSYBOX_ARCH}" >/dev/null 2>&1 && \
|
|
|
|
make -j"$(nproc)" >/dev/null 2>&1 && \
|
|
|
|
make -j"$(nproc)" \
|
|
|
|
strip busybox
|
|
|
|
ARCH="${BUSYBOX_ARCH}" \
|
|
|
|
|
|
|
|
CROSS_COMPILE="${CROSS}-" \
|
|
|
|
|
|
|
|
CC="${CROSS}-clang" \
|
|
|
|
|
|
|
|
HOSTCC=gcc \
|
|
|
|
|
|
|
|
SKIP_STRIP=y \
|
|
|
|
|
|
|
|
>/dev/null 2>&1 && \
|
|
|
|
|
|
|
|
llvm-strip busybox
|
|
|
|
|
|
|
|
|
|
|
|
# Lay out a minimal rootfs with busybox + an applet symlink per applet.
|
|
|
|
# Lay out a minimal rootfs with busybox + an applet symlink per applet.
|
|
|
|
# Symlinks (argv[0] dispatch) are how busybox selects an applet and make the
|
|
|
|
# Symlinks (argv[0] dispatch) are how busybox selects an applet and make the
|
|
|
@@ -290,6 +366,10 @@ RUN make allnoconfig && \
|
|
|
|
# for non-applet symbols like FEATURE_* / STATIC, which we filter out).
|
|
|
|
# for non-applet symbols like FEATURE_* / STATIC, which we filter out).
|
|
|
|
# We generate symlinks from the UNCOMPRESSED binary (so the probe is reliable),
|
|
|
|
# We generate symlinks from the UNCOMPRESSED binary (so the probe is reliable),
|
|
|
|
# then UPX-compress the binary in place afterwards.
|
|
|
|
# then UPX-compress the binary in place afterwards.
|
|
|
|
|
|
|
|
#
|
|
|
|
|
|
|
|
# Note: probing cross-compiled binaries requires binfmt/QEMU user-mode. This
|
|
|
|
|
|
|
|
# is only a lightweight per-applet help-flag check, not a full emulated build.
|
|
|
|
|
|
|
|
# If QEMU is unavailable in CI, replace the probe with a static applet list.
|
|
|
|
RUN mkdir -p /rootfs/bin && \
|
|
|
|
RUN mkdir -p /rootfs/bin && \
|
|
|
|
grep '^CONFIG_.*=y' .config \
|
|
|
|
grep '^CONFIG_.*=y' .config \
|
|
|
|
| sed -e 's/^CONFIG_//' -e 's/=y$//' \
|
|
|
|
| sed -e 's/^CONFIG_//' -e 's/=y$//' \
|
|
|
|