commit:     4605fd652f147c1e60c6675d122fba07e20a9d71
Author:     Sam James <sam <AT> gentoo <DOT> org>
AuthorDate: Thu Feb  5 03:49:18 2026 +0000
Commit:     Sam James <sam <AT> gentoo <DOT> org>
CommitDate: Thu Feb  5 03:51:49 2026 +0000
URL:        https://gitweb.gentoo.org/repo/gentoo.git/commit/?id=4605fd65

sys-libs/musl: backport further patches, including a printf one for systemd

I took a look at what Alpine is using and picked up some more patches. Was
considering a snapshot like in 2ba8dfe9407383b97485385552cab675a51b8bd5 but
Alpine had some seccomp issues (f2036bfe000ab05e1f300c6fce57be3da9db426e
in aports.git), maybe we should just do that though. The patch stack is
getting a bit much.

Signed-off-by: Sam James <sam <AT> gentoo.org>

 sys-libs/musl/files/musl-isatty.patch             |  40 ++++
 sys-libs/musl/files/musl-page-size.patch          |  48 ++++
 sys-libs/musl/files/musl-printf-empty-iovec.patch |  33 +++
 sys-libs/musl/musl-1.2.5-r8.ebuild                | 267 ++++++++++++++++++++++
 4 files changed, 388 insertions(+)

diff --git a/sys-libs/musl/files/musl-isatty.patch 
b/sys-libs/musl/files/musl-isatty.patch
new file mode 100644
index 000000000000..df57e5a01e53
--- /dev/null
+++ b/sys-libs/musl/files/musl-isatty.patch
@@ -0,0 +1,40 @@
+https://git.musl-libc.org/cgit/musl/commit/?id=c94a0c16f08894ce3be6dafb0fe80baa77a6ff2a
+
+From c94a0c16f08894ce3be6dafb0fe80baa77a6ff2a Mon Sep 17 00:00:00 2001
+From: Rich Felker <[email protected]>
+Date: Tue, 20 Aug 2024 12:34:50 -0400
+Subject: isatty: don't collapse all non-EBADF errors to ENOTTY
+
+linux puts hung-up ttys in a state where ioctls produce EIO, and may
+do the same for other types of devices in error or shutdown states.
+such an error clearly does not mean the device is not a tty, but it
+also can't reliably establish that the device is a tty, so the only
+safe thing to do seems to be reporting the error. programs that don't
+check errno will conclude that the device is not a tty, which is no
+different from what happens now, but at least they gain the option to
+differentiate between the cases.
+
+commit c84971995b3a6d5118f9357c040572f4c78bcd55 introduced the errno
+collapsing behavior, but prior to that, errno was not set at all by
+isatty.
+---
+ src/unistd/isatty.c | 6 ++----
+ 1 file changed, 2 insertions(+), 4 deletions(-)
+
+diff --git a/src/unistd/isatty.c b/src/unistd/isatty.c
+index 75a9c186..21222eda 100644
+--- a/src/unistd/isatty.c
++++ b/src/unistd/isatty.c
+@@ -6,8 +6,6 @@
+ int isatty(int fd)
+ {
+       struct winsize wsz;
+-      unsigned long r = syscall(SYS_ioctl, fd, TIOCGWINSZ, &wsz);
+-      if (r == 0) return 1;
+-      if (errno != EBADF) errno = ENOTTY;
+-      return 0;
++      /* +1 converts from error status (0/-1) to boolean (1/0) */
++      return syscall(SYS_ioctl, fd, TIOCGWINSZ, &wsz) + 1;
+ }
+-- 
+cgit v1.2.1

diff --git a/sys-libs/musl/files/musl-page-size.patch 
b/sys-libs/musl/files/musl-page-size.patch
new file mode 100644
index 000000000000..830500c25340
--- /dev/null
+++ b/sys-libs/musl/files/musl-page-size.patch
@@ -0,0 +1,48 @@
+https://git.musl-libc.org/cgit/musl/commit/?id=6f666231bf51703fadbef10460d462fb573548a1
+
+From 6f666231bf51703fadbef10460d462fb573548a1 Mon Sep 17 00:00:00 2001
+From: Rich Felker <[email protected]>
+Date: Tue, 7 May 2024 08:18:49 -0400
+Subject: ldso: fix non-functional fix to early dynamic PAGE_SIZE access
+
+commit f47a8cdd250d9163fcfb39bf4e9d813957c0b187 introduced an
+alternate mechanism for access to runtime page size for compatibility
+with early stages of dynamic linking, but because pthread_impl.h
+indirectly includes libc.h, the condition #ifndef PAGE_SIZE was never
+satisfied.
+
+rather than depend on order of inclusion, use the (baseline POSIX)
+macro PAGESIZE, not the (XSI) macro PAGE_SIZE, to determine whether
+page size is dynamic. our internal libc.h only provides a dynamic
+definition for PAGE_SIZE, not for PAGESIZE.
+---
+ ldso/dynlink.c | 8 +++++---
+ 1 file changed, 5 insertions(+), 3 deletions(-)
+
+diff --git a/ldso/dynlink.c b/ldso/dynlink.c
+index 324aa859..42687da2 100644
+--- a/ldso/dynlink.c
++++ b/ldso/dynlink.c
+@@ -21,15 +21,17 @@
+ #include <sys/membarrier.h>
+ #include "pthread_impl.h"
+ #include "fork_impl.h"
++#include "libc.h"
+ #include "dynlink.h"
+ 
+ static size_t ldso_page_size;
+-#ifndef PAGE_SIZE
++/* libc.h may have defined a macro for dynamic PAGE_SIZE already, but
++ * PAGESIZE is only defined if it's constant for the arch. */
++#ifndef PAGESIZE
++#undef PAGE_SIZE
+ #define PAGE_SIZE ldso_page_size
+ #endif
+ 
+-#include "libc.h"
+-
+ #define malloc __libc_malloc
+ #define calloc __libc_calloc
+ #define realloc __libc_realloc
+-- 
+cgit v1.2.1

diff --git a/sys-libs/musl/files/musl-printf-empty-iovec.patch 
b/sys-libs/musl/files/musl-printf-empty-iovec.patch
new file mode 100644
index 000000000000..8cfd52c8a034
--- /dev/null
+++ b/sys-libs/musl/files/musl-printf-empty-iovec.patch
@@ -0,0 +1,33 @@
+https://github.com/systemd/systemd/pull/38825#issue-3386123778
+https://git.musl-libc.org/cgit/musl/commit/?id=fde29c04adbab9d5b081bf6717b5458188647f1c
+
+From fde29c04adbab9d5b081bf6717b5458188647f1c Mon Sep 17 00:00:00 2001
+From: Casey Connolly <[email protected]>
+Date: Wed, 23 Apr 2025 15:06:48 +0200
+Subject: stdio: skip empty iovec when buffering is disabled
+
+When buffering on a FILE is disabled we still send both iovecs, even
+though the first one is always empty. Clean things up by skipping the
+empty iovec instead.
+---
+ src/stdio/__stdio_write.c | 5 +++++
+ 1 file changed, 5 insertions(+)
+
+diff --git a/src/stdio/__stdio_write.c b/src/stdio/__stdio_write.c
+index d2d89475..5356553d 100644
+--- a/src/stdio/__stdio_write.c
++++ b/src/stdio/__stdio_write.c
+@@ -11,6 +11,11 @@ size_t __stdio_write(FILE *f, const unsigned char *buf, 
size_t len)
+       size_t rem = iov[0].iov_len + iov[1].iov_len;
+       int iovcnt = 2;
+       ssize_t cnt;
++
++      if (!iov->iov_len) {
++              iov++;
++              iovcnt--;
++      }
+       for (;;) {
+               cnt = syscall(SYS_writev, f->fd, iov, iovcnt);
+               if (cnt == rem) {
+-- 
+cgit v1.2.1

diff --git a/sys-libs/musl/musl-1.2.5-r8.ebuild 
b/sys-libs/musl/musl-1.2.5-r8.ebuild
new file mode 100644
index 000000000000..4dfc2184e140
--- /dev/null
+++ b/sys-libs/musl/musl-1.2.5-r8.ebuild
@@ -0,0 +1,267 @@
+# Copyright 1999-2026 Gentoo Authors
+# Distributed under the terms of the GNU General Public License v2
+
+EAPI=8
+
+inherit crossdev flag-o-matic toolchain-funcs prefix
+
+DESCRIPTION="Light, fast and, simple C library focused on 
standards-conformance and safety"
+HOMEPAGE="https://musl.libc.org";
+
+if [[ ${PV} == 9999 ]] ; then
+       EGIT_REPO_URI="https://git.musl-libc.org/git/musl";
+       inherit git-r3
+else
+       VERIFY_SIG_OPENPGP_KEY_PATH=/usr/share/openpgp-keys/musl.asc
+       inherit verify-sig
+
+       SRC_URI="https://musl.libc.org/releases/${P}.tar.gz";
+       SRC_URI+=" verify-sig? ( https://musl.libc.org/releases/${P}.tar.gz.asc 
)"
+       KEYWORDS="-* ~amd64 ~arm ~arm64 ~m68k ~mips ~ppc ~ppc64 ~riscv ~x86"
+
+       BDEPEND="verify-sig? ( sec-keys/openpgp-keys-musl )"
+fi
+
+GETENT_COMMIT="93a08815f8598db442d8b766b463d0150ed8e2ab"
+GETENT_FILE="musl-getent-${GETENT_COMMIT}.c"
+SRC_URI+="
+       https://dev.gentoo.org/~blueness/musl-misc/getconf.c
+       
https://gitlab.alpinelinux.org/alpine/aports/-/raw/${GETENT_COMMIT}/main/musl/getent.c
 -> ${GETENT_FILE}
+       https://dev.gentoo.org/~blueness/musl-misc/iconv.c
+"
+
+LICENSE="MIT LGPL-2 GPL-2"
+SLOT="0"
+IUSE="crypt headers-only split-usr"
+
+QA_SONAME="usr/lib/libc.so"
+QA_DT_NEEDED="usr/lib/libc.so"
+# bug #830213
+QA_PRESTRIPPED="usr/lib/crtn.o"
+
+# We want crypt on by default for this as sys-libs/libxcrypt isn't (yet?)
+# built as part as crossdev. Also, elide the blockers when in cross-*,
+# as it doesn't make sense to block the normal CBUILD libxcrypt at all
+# there when we're installing into /usr/${CHOST} anyway.
+if is_crosspkg ; then
+       IUSE="${IUSE/crypt/+crypt}"
+else
+       RDEPEND="crypt? ( !sys-libs/libxcrypt[system] )"
+       PDEPEND="!crypt? ( sys-libs/libxcrypt[system] )"
+fi
+
+PATCHES=(
+       "${FILESDIR}"/${PN}-1.2.4-arm64-crti-alignment.patch
+       "${FILESDIR}"/${PN}-sched.h-reduce-namespace-conflicts.patch
+       "${FILESDIR}"/${PN}-iconv-out-of-bound-fix.patch
+       "${FILESDIR}"/${PN}-arm-crti-alignment.patch
+       "${FILESDIR}"/${PN}-ppc-clobber.patch
+       "${FILESDIR}"/${PN}-dns-union.patch
+       "${FILESDIR}"/${PN}-getauxval.patch
+       "${FILESDIR}"/${PN}-getifaddrs-qemu-workaround.patch
+       "${FILESDIR}"/${PN}-page-size.patch
+       "${FILESDIR}"/${PN}-isatty.patch
+       "${FILESDIR}"/${PN}-printf-empty-iovec.patch
+)
+
+just_headers() {
+       use headers-only && target_is_not_host
+}
+
+pkg_setup() {
+       if [[ ${CTARGET} == ${CHOST} ]] ; then
+               case ${CHOST} in
+                       *-musl*) ;;
+                       *) die "Use sys-devel/crossdev to build a musl 
toolchain" ;;
+               esac
+       fi
+
+       # Fix for bug #667126, copied from glibc ebuild:
+       # make sure host make.conf doesn't pollute us
+       if target_is_not_host || tc-is-cross-compiler ; then
+               CHOST=${CTARGET} strip-unsupported-flags
+       fi
+}
+
+src_unpack() {
+       if [[ ${PV} == 9999 ]] ; then
+               git-r3_src_unpack
+       elif use verify-sig ; then
+               # We only verify the release; not the additional (fixed, safe) 
files
+               # we download.
+               # (Seem to get IPC error on verifying in cross?)
+               ! target_is_not_host && verify-sig_verify_detached 
"${DISTDIR}"/${P}.tar.gz{,.asc}
+       fi
+
+       default
+}
+
+src_prepare() {
+       default
+
+       mkdir "${WORKDIR}"/misc || die
+       cp "${DISTDIR}"/getconf.c "${WORKDIR}"/misc/getconf.c || die
+       cp "${DISTDIR}/${GETENT_FILE}" "${WORKDIR}"/misc/getent.c || die
+       cp "${DISTDIR}"/iconv.c "${WORKDIR}"/misc/iconv.c || die
+}
+
+src_configure() {
+       strip-flags && filter-lto # Prevent issues caused by aggressive 
optimizations & bug #877343
+       tc-getCC ${CTARGET}
+
+       just_headers && export CC=true
+
+       local sysroot
+       target_is_not_host && sysroot=/usr/${CTARGET}
+       ./configure \
+               --target=${CTARGET} \
+               --prefix="${EPREFIX}${sysroot}/usr" \
+               --syslibdir="${EPREFIX}${sysroot}/lib" \
+               --disable-gcc-wrapper || die
+}
+
+src_compile() {
+       emake obj/include/bits/alltypes.h
+       just_headers && return 0
+
+       emake
+       if ! is_crosspkg ; then
+               emake -C "${T}" getconf getent iconv \
+                       CC="$(tc-getCC)" \
+                       CFLAGS="${CFLAGS}" \
+                       CPPFLAGS="${CPPFLAGS}" \
+                       LDFLAGS="${LDFLAGS}" \
+                       VPATH="${WORKDIR}/misc"
+       fi
+
+       $(tc-getCC) ${CPPFLAGS} ${CFLAGS} -c -o libssp_nonshared.o 
"${FILESDIR}"/stack_chk_fail_local.c || die
+       $(tc-getAR) -rcs libssp_nonshared.a libssp_nonshared.o || die
+}
+
+src_install() {
+       local target="install"
+       just_headers && target="install-headers"
+       emake DESTDIR="${D}" ${target}
+       just_headers && return 0
+
+       # musl provides ldd via a sym link to its ld.so
+       local sysroot=
+       target_is_not_host && sysroot=/usr/${CTARGET}
+       local ldso=$(basename "${ED}${sysroot}"/lib/ld-musl-*)
+       dosym -r "${sysroot}/lib/${ldso}" "${sysroot}/usr/bin/ldd"
+
+       if ! use crypt ; then
+               # Allow sys-libs/libxcrypt[system] to provide it instead
+               rm "${ED}${sysroot}/usr/include/crypt.h" || die
+               rm "${ED}${sysroot}"/usr/*/libcrypt.a || die
+       fi
+
+       if ! is_crosspkg ; then
+               # Fish out of config:
+               #   ARCH = ...
+               #   SUBARCH = ...
+               # and print $(ARCH)$(SUBARCH).
+               local arch=$(awk '{ k[$1] = $3 } END { printf("%s%s", 
k["ARCH"], k["SUBARCH"]); }' config.mak)
+
+               # The musl build system seems to create a symlink:
+               # ${D}/lib/ld-musl-${arch}.so.1 -> /usr/lib/libc.so.1 (absolute)
+               # During cross or within prefix, there's no guarantee that the 
host is
+               # using musl so that file may not exist. Use a relative symlink 
within
+               # ${D} instead.
+               rm "${ED}"/lib/ld-musl-${arch}.so.1 || die
+               if use split-usr; then
+                       dosym ../usr/lib/libc.so /lib/ld-musl-${arch}.so.1
+                       # If it's still a dead symlink, OK, we really do need 
to abort.
+                       [[ -e "${ED}"/lib/ld-musl-${arch}.so.1 ]] || die
+               else
+                       dosym libc.so /usr/lib/ld-musl-${arch}.so.1
+                       [[ -e "${ED}"/usr/lib/ld-musl-${arch}.so.1 ]] || die
+               fi
+
+               cp "${FILESDIR}"/ldconfig.in-r3 "${T}"/ldconfig.in || die
+               sed -e "s|@@ARCH@@|${arch}|" "${T}"/ldconfig.in > 
"${T}"/ldconfig || die
+               eprefixify "${T}"/ldconfig
+               into /
+               dosbin "${T}"/ldconfig
+               into /usr
+               dobin "${T}"/getconf
+               dobin "${T}"/getent
+               dobin "${T}"/iconv
+               newenvd - "00musl" <<-EOF
+               # 00musl autogenerated by sys-libs/musl ebuild; DO NOT EDIT.
+               LDPATH="include ld.so.conf.d/*.conf"
+               EOF
+       fi
+
+       if target_is_not_host ; then
+               into /usr/${CTARGET}
+               dolib.a libssp_nonshared.a
+       else
+               dolib.a libssp_nonshared.a
+       fi
+}
+
+# Simple test to make sure our new musl isn't completely broken.
+# Make sure we don't test with statically built binaries since
+# they will fail.  Also, skip if this musl is a cross compiler.
+#
+# If coreutils is built with USE=multicall, some of these files
+# will just be wrapper scripts, not actual ELFs we can test.
+musl_sanity_check() {
+       cd / #228809
+
+       # We enter ${ED} so to avoid trouble if the path contains
+       # special characters; for instance if the path contains the
+       # colon character (:), then the linker will try to split it
+       # and look for the libraries in an unexpected place. This can
+       # lead to unsafe code execution if the generated prefix is
+       # within a world-writable directory.
+       # (e.g. /var/tmp/portage:${HOSTNAME})
+       pushd "${ED}"/usr/$(get_libdir) >/dev/null
+
+       # first let's find the actual dynamic linker here
+       # symlinks may point to the wrong abi
+       local newldso=$(find . -maxdepth 1 -name 'libc.so' -type f -print -quit)
+
+       einfo Last-minute run tests with ${newldso} in /usr/$(get_libdir) ...
+
+       local x striptest
+       for x in cal date env free ls true uname uptime ; do
+               x=$(type -p ${x})
+               [[ -z ${x} || ${x} != ${EPREFIX}/* ]] && continue
+               striptest=$(LC_ALL="C" file -L ${x} 2>/dev/null) || continue
+               case ${striptest} in
+               *"statically linked"*) continue;;
+               *"static-pie linked"*) continue;;
+               *"ASCII text"*) continue;;
+               esac
+               # We need to clear the locale settings as the upgrade might want
+               # incompatible locale data.  This test is not for verifying 
that.
+               LC_ALL=C \
+               ${newldso} --library-path . ${x} > /dev/null \
+                       || die "simple run test (${x}) failed"
+       done
+
+       popd >/dev/null
+}
+
+pkg_preinst() {
+       # Nothing to do if just installing headers
+       just_headers && return
+
+       # Prepare /etc/ld.so.conf.d/ for files
+       mkdir -p "${EROOT}"/etc/ld.so.conf.d
+
+       [[ -n ${ROOT} ]] && return 0
+       [[ -d ${ED}/usr/$(get_libdir) ]] || return 0
+       target_is_not_host && return 0
+       musl_sanity_check
+}
+
+pkg_postinst() {
+       target_is_not_host && return 0
+
+       [[ -n "${ROOT}" ]] && return 0
+
+       ldconfig || die
+}

Reply via email to