GNU Coreutils, BusyBox, and Toybox buffer the input of the base*
commands so that they can operate on files that do not fit into memory,
but uutils does not:
$ s=10000000 # 10GiB
$ timeout 10 sh -c \
"(ulimit -v $s && base64 < /dev/zero >out 2>err)"
$ cat err && wc -c out
6514348032 out
$ timeout 10 sh -c \
"(ulimit -v $s && busybox base64 < /dev/zero >out 2>err)"
$ cat err && wc -c out
1189785519 out
$ timeout 10 sh -c \
"(ulimit -v $s && toybox base64 < /dev/zero >out 2>err)"
$ cat err && wc -c out
199167661 out
$ timeout 10 sh -c \
"(ulimit -v $s && uutils base64 < /dev/zero >out 2>err)"
$ cat err && wc -c out
base64: out of memory
0 out
* tests/basenc/basenc-bounded-memory.sh: New file.
* tests/local.mk (all_tests): Add the test.
---
tests/basenc/basenc-bounded-memory.sh | 41 +++++++++++++++++++++++++++
tests/local.mk | 1 +
2 files changed, 42 insertions(+)
create mode 100755 tests/basenc/basenc-bounded-memory.sh
diff --git a/tests/basenc/basenc-bounded-memory.sh
b/tests/basenc/basenc-bounded-memory.sh
new file mode 100755
index 000000000..6bd7f6e35
--- /dev/null
+++ b/tests/basenc/basenc-bounded-memory.sh
@@ -0,0 +1,41 @@
+#!/bin/sh
+# Test that basenc can work on files larger than the systems memory.
+
+# Copyright (C) 2025 Free Software Foundation, Inc.
+
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+
+# You should have received a copy of the GNU General Public License
+# along with this program. If not, see <https://www.gnu.org/licenses/>.
+
+. "${srcdir=.}/tests/init.sh"; path_prepend_ ./src
+print_ver_ basenc
+
+test -w /dev/full && test -c /dev/full || skip_ 'this system lacks /dev/full'
+
+vm=$(get_min_ulimit_v_ basenc /dev/null)
+
+# Check all except for --base58.
+for algorithm in '--base64' '--base64url' '--base32' '--base32hex' '--base16' \
+ '--base2msbf' '--base2lsbf' '--z85'; do
+ timeout 2 $SHELL -c \
+ "(ulimit -v $(($vm+24000)) \
+ && basenc $algorithm </dev/full >/dev/null 2>err)"
+ ret=$?
+ test -f err || skip_ 'shell ulimit failure'
+ test $ret = 124 || {
+ fail=1
+ cat err
+ echo "basenc $algorithm used too much memory" >&2
+ }
+done
+
+Exit $fail
diff --git a/tests/local.mk b/tests/local.mk
index 52184b7ac..4c02cc65d 100644
--- a/tests/local.mk
+++ b/tests/local.mk
@@ -298,6 +298,7 @@ all_tests = \
tests/misc/basename.pl \
tests/basenc/base64.pl \
tests/basenc/basenc.pl \
+ tests/basenc/basenc-bounded-memory.sh \
tests/basenc/basenc-large.sh \
tests/misc/close-stdout.sh \
tests/chroot/chroot-fail.sh \
--
2.51.0