2017-12-24 09:53:20 +03:00
#!/bin/sh
2021-10-01 14:04:32 +03:00
# SPDX-License-Identifier: LGPL-2.1-or-later
2020-12-06 19:45:01 +03:00
set -e
2016-07-16 03:00:44 +03:00
# This is a build script for OS image generation using mkosi (https://github.com/systemd/mkosi).
# Simply invoke "mkosi" in the project directory to build an OS image.
2022-07-15 03:26:52 +03:00
ASAN_OPTIONS=strict_string_checks=1:detect_stack_use_after_return=1:check_initialization_order=1:strict_init_order=1:disable_coredump=0:use_madv_dontdump=1
UBSAN_OPTIONS=print_stacktrace=1:print_summary=1:halt_on_error=1
2021-04-01 18:58:54 +03:00
# On Fedora "ld" is (unfortunately — if you ask me) managed via
# "alternatives". Since we'd like to support building images in environments
# with only /usr/ around (e.g. mkosi's UsrOnly=1 option), we have the problem
# that /usr/bin/ld is a symlink that points to a non-existing file in
# /etc/alternative/ in this mode. Let's work around this for now by manually
# redirect "ld" to "ld.bfd", i.e. circumventing the /usr/bin/ld symlink.
2021-09-02 15:22:19 +03:00
if [ ! -x /usr/bin/ld ] && [ -x /usr/bin/ld.bfd ]; then
2021-04-01 18:58:54 +03:00
mkdir -p "$HOME"/bin
ln -s /usr/bin/ld.bfd "$HOME"/bin/ld
PATH="$HOME/bin:$PATH"
fi
2017-07-12 20:58:53 +03:00
# If mkosi.builddir/ exists mkosi will set $BUILDDIR to it, let's then use it
# as out-of-tree build dir. Otherwise, let's make up our own builddir.
[ -z "$BUILDDIR" ] && BUILDDIR=build
2018-03-05 19:03:52 +03:00
# Meson uses Python 3 and requires a locale with an UTF-8 character map.
# Not running under UTF-8 makes the `ninja test` step break with a CodecError.
# So let's ensure we're running under UTF-8.
#
# If our current locale already is UTF-8, then we don't need to do anything:
2021-09-02 15:22:19 +03:00
if [ "$(locale charmap 2>/dev/null)" != "UTF-8" ] ; then
2018-03-05 19:03:52 +03:00
# Try using C.UTF-8 locale, if available. This locale is not shipped
# by upstream glibc, so it's not available in all distros.
# (In particular, it's not available in Arch Linux.)
2020-12-06 21:16:00 +03:00
if locale -a | grep -q -E "C.UTF-8|C.utf8"; then
export LC_CTYPE=C.UTF-8
# Finally, try something like en_US.UTF-8, which should be
# available in Arch Linux, but is not present in Debian's
# minimal image in our mkosi config.
elif locale -a | grep -q en_US.utf8; then
2018-03-05 19:03:52 +03:00
export LC_CTYPE=en_US.UTF-8
2020-12-06 21:16:00 +03:00
else
# If nothing works, fail early.
echo "*** Could not find a valid locale that supports UTF-8. ***" >&2
exit 1
2018-03-05 19:03:52 +03:00
fi
fi
2017-07-18 11:30:52 +03:00
2018-02-09 21:22:40 +03:00
if [ ! -f "$BUILDDIR"/build.ninja ] ; then
2021-09-02 15:22:19 +03:00
sysvinit_path=$(realpath /etc/init.d)
2017-10-25 21:42:38 +03:00
2021-09-02 15:22:19 +03:00
init_path=$(realpath /sbin/init 2>/dev/null)
2020-04-21 19:08:23 +03:00
if [ -z "$init_path" ] ; then
2022-07-16 11:15:29 +03:00
rootprefix=""
2020-04-21 19:08:23 +03:00
else
2022-07-16 11:15:29 +03:00
rootprefix=${init_path%/lib/systemd/systemd}
rootprefix=/${rootprefix#/}
2020-04-21 19:08:23 +03:00
fi
2021-01-09 22:05:56 +03:00
meson "$BUILDDIR" \
-D "sysvinit-path=$sysvinit_path" \
-D "rootprefix=$rootprefix" \
-D man=false \
2021-01-15 22:51:42 +03:00
-D translations=false \
2022-07-15 15:59:03 +03:00
-D version-tag="${VERSION_TAG}" \
2022-07-15 03:26:52 +03:00
-D mode=developer \
-D b_sanitize="${SANITIZERS:-none}"
2017-12-06 15:53:39 +03:00
fi
2021-01-09 19:32:00 +03:00
cd "$BUILDDIR"
2021-01-24 17:13:58 +03:00
ninja "$@"
2020-04-22 02:58:44 +03:00
if [ "$WITH_TESTS" = 1 ] ; then
for id in 1 2 3; do
2021-09-02 15:22:19 +03:00
getent group $id >/dev/null || groupadd -g $id testgroup$id
2020-04-22 02:58:44 +03:00
done
2022-07-15 03:26:52 +03:00
if [ -n "$SANITIZERS" ]; then
export ASAN_OPTIONS="$ASAN_OPTIONS"
export UBSAN_OPTIONS="$UBSAN_OPTIONS"
TIMEOUT_MULTIPLIER=3
else
TIMEOUT_MULTIPLIER=1
fi
meson test --timeout-multiplier=$TIMEOUT_MULTIPLIER
2020-04-22 02:58:44 +03:00
fi
2021-01-09 19:32:00 +03:00
cd "$SRCDIR"
2020-12-06 19:45:45 +03:00
# Ubuntu Focal is stuck with meson 0.53.0.
if [ "$(meson -v | cut -d . -f 2)" -gt 53 ] ; then
2020-12-19 17:25:41 +03:00
meson install -C "$BUILDDIR" --quiet --no-rebuild --only-changed
2020-12-06 19:45:45 +03:00
else
2020-12-19 17:25:41 +03:00
meson install -C "$BUILDDIR" --no-rebuild --only-changed
2020-12-06 19:45:45 +03:00
fi
2016-12-07 01:17:57 +03:00
2017-07-12 20:58:53 +03:00
mkdir -p "$DESTDIR"/etc
2016-12-07 01:17:57 +03:00
2021-09-02 15:22:19 +03:00
cat >"$DESTDIR"/etc/issue <<EOF
2016-12-07 01:17:57 +03:00
\S (built from systemd tree)
Kernel \r on an \m (\l)
EOF
2018-06-21 19:47:33 +03:00
2021-07-02 15:59:58 +03:00
if [ -n "$IMAGE_ID" ] ; then
2022-07-16 11:15:29 +03:00
mkdir -p "$DESTDIR"/usr/lib
sed -n \
-e '/^IMAGE_ID=/!p' \
-e "\$aIMAGE_ID=$IMAGE_ID" <"/usr/lib/os-release" >"${DESTDIR}/usr/lib/os-release"
2021-07-02 15:59:58 +03:00
2022-07-16 11:15:29 +03:00
OSRELEASEFILE="$DESTDIR"/usr/lib/os-release
2021-07-02 15:59:58 +03:00
else
2022-07-16 11:15:29 +03:00
OSRELEASEFILE=/usr/lib/os-release
2021-07-02 15:59:58 +03:00
fi
if [ -n "$IMAGE_VERSION" ] ; then
2022-07-16 11:15:29 +03:00
mkdir -p "$DESTDIR"/usr/lib
sed -n \
-e '/^IMAGE_VERSION=/!p' \
-e "\$aIMAGE_VERSION=$IMAGE_VERSION" <$OSRELEASEFILE >"/tmp/os-release.tmp"
2021-07-02 15:59:58 +03:00
2022-07-16 11:15:29 +03:00
cat /tmp/os-release.tmp > "$DESTDIR"/usr/lib/os-release
rm /tmp/os-release.tmp
2021-07-02 15:59:58 +03:00
fi
2021-12-08 21:31:25 +03:00
# If $CI_BUILD is set, copy over the CI service which executes a service check
# after boot and then shuts down the machine
if [ -n "$CI_BUILD" ]; then
2022-07-16 11:15:29 +03:00
mkdir -p "$DESTDIR/usr/lib/systemd/system"
cp -v "$SRCDIR/test/mkosi-check-and-shutdown.service" "$DESTDIR/usr/lib/systemd/system/mkosi-check-and-shutdown.service"
cp -v "$SRCDIR/test/mkosi-check-and-shutdown.sh" "$DESTDIR/usr/lib/systemd/mkosi-check-and-shutdown.sh"
chmod +x "$DESTDIR/usr/lib/systemd/mkosi-check-and-shutdown.sh"
2021-12-08 21:31:25 +03:00
fi
2022-07-15 03:26:52 +03:00
if [ -n "$SANITIZERS" ]; then
LD_PRELOAD=$(ldd $BUILDDIR/systemd | grep libasan.so | awk '{print $3}')
mkdir -p "$DESTDIR/etc/systemd/system.conf.d"
cat > "$DESTDIR/etc/systemd/system.conf.d/10-asan.conf" <<EOF
[Manager]
ManagerEnvironment=ASAN_OPTIONS=$ASAN_OPTIONS\\
UBSAN_OPTIONS=$UBSAN_OPTIONS\\
LD_PRELOAD=$LD_PRELOAD
DefaultEnvironment=ASAN_OPTIONS=$ASAN_OPTIONS\\
UBSAN_OPTIONS=$UBSAN_OPTIONS\\
LD_PRELOAD=$LD_PRELOAD
EOF
# ASAN logs to stderr by default. However, journald's stderr is connected to /dev/null, so we lose
# all the ASAN logs. To rectify that, let's connect journald's stdout to the console so that any
# sanitizer failures appear directly on the user's console.
mkdir -p "$DESTDIR/etc/systemd/system/systemd-journald.service.d"
cat > "$DESTDIR/etc/systemd/system/systemd-journald.service.d/10-stdout-tty.conf" <<EOF
[Service]
StandardOutput=tty
EOF
# Both systemd and util-linux's login call vhangup() on /dev/console which disconnects all users.
# This means systemd-journald can't log to /dev/console even if we configure `StandardOutput=tty`. As
# a workaround, we modify console-getty.service to disable systemd's vhangup() and disallow login
# from calling vhangup() so that journald's ASAN logs correctly end up in the console.
mkdir -p "$DESTDIR/etc/systemd/system/console-getty.service.d"
cat > "$DESTDIR/etc/systemd/system/console-getty.service.d/10-no-vhangup.conf" <<EOF
[Service]
TTYVHangup=no
CapabilityBoundingSet=~CAP_SYS_TTY_CONFIG
EOF
fi