2021-08-19 16:54:40 +02:00
#!/usr/bin/env bash
# SPDX-License-Identifier: LGPL-2.1-or-later
# -*- mode: shell-script; indent-tabs-mode: nil; sh-basic-offset: 4; -*-
# ex: ts=8 sw=4 sts=4 et filetype=sh
set -eux
set -o pipefail
SYSUPDATE = /lib/systemd/systemd-sysupdate
2024-03-27 20:13:24 +01:00
SECTOR_SIZES = ( 512 4096)
WORKDIR = " $( mktemp -d /var/tmp/test-72-XXXXXX) "
BACKING_FILE = " $WORKDIR /joined.raw "
export SYSTEMD_ESP_PATH = " $WORKDIR /esp "
export SYSTEMD_XBOOTLDR_PATH = " $WORKDIR /xbootldr "
2023-08-11 22:05:45 +10:00
export SYSTEMD_PAGER = cat
export SYSTEMD_LOG_LEVEL = debug
2021-08-19 16:54:40 +02:00
2024-03-27 20:13:24 +01:00
if [ [ ! -x " $SYSUPDATE " ] ] ; then
2021-08-19 16:54:40 +02:00
echo "no systemd-sysupdate" >/skipped
2024-04-02 20:37:30 +02:00
exit 77
2021-08-19 16:54:40 +02:00
fi
2023-08-11 22:05:45 +10:00
# Loopback devices may not be supported. They are used because sfdisk cannot
# change the sector size of a file, and we want to test both 512 and 4096 byte
# sectors. If loopback devices are not supported, we can only test one sector
# size, and the underlying device is likely to have a sector size of 512 bytes.
2024-03-27 19:35:30 +01:00
if [ [ ! -e /dev/loop-control ] ] ; then
2023-08-11 22:05:45 +10:00
echo "No loopback device support"
2024-03-27 20:13:24 +01:00
SECTOR_SIZES = ( 512)
2023-08-11 22:05:45 +10:00
fi
2021-08-19 16:54:40 +02:00
2024-03-27 20:13:24 +01:00
at_exit( ) {
set +e
losetup -n --output NAME --associated " $BACKING_FILE " | while read -r loop_dev; do
losetup --detach " $loop_dev "
done
rm -rf " $WORKDIR "
2023-08-11 22:05:45 +10:00
}
2024-03-27 20:13:24 +01:00
trap at_exit EXIT
2023-08-11 22:05:45 +10:00
new_version( ) {
2024-03-27 20:13:24 +01:00
local sector_size = " ${ 1 : ? } "
local version = " ${ 2 : ? } "
2023-08-11 22:05:45 +10:00
# Create a pair of random partition payloads, and compress one
2024-03-27 20:13:24 +01:00
dd if = /dev/urandom of = " $WORKDIR /source/part1- $version .raw " bs = " $sector_size " count = 2048
dd if = /dev/urandom of = " $WORKDIR /source/part2- $version .raw " bs = " $sector_size " count = 2048
gzip -k -f " $WORKDIR /source/part2- $version .raw "
2023-08-11 22:05:45 +10:00
# Create a random "UKI" payload
2024-03-27 20:13:24 +01:00
echo $RANDOM >" $WORKDIR /source/uki- $version .efi "
2023-08-11 22:05:45 +10:00
2023-09-24 14:35:59 +02:00
# Create a random extra payload
2024-03-27 20:13:24 +01:00
echo $RANDOM >" $WORKDIR /source/uki-extra- $version .efi "
2023-09-24 14:35:59 +02:00
2023-08-11 22:05:45 +10:00
# Create tarball of a directory
2024-03-27 20:13:24 +01:00
mkdir -p " $WORKDIR /source/dir- $version "
echo $RANDOM >" $WORKDIR /source/dir- $version /foo.txt "
echo $RANDOM >" $WORKDIR /source/dir- $version /bar.txt "
tar --numeric-owner -C " $WORKDIR /source/dir- $version / " -czf " $WORKDIR /source/dir- $version .tar.gz " .
2023-08-11 22:05:45 +10:00
2024-03-27 20:13:24 +01:00
( cd " $WORKDIR /source " && sha256sum uki* part* dir-*.tar.gz >SHA256SUMS)
2023-08-11 22:05:45 +10:00
}
2021-08-19 16:54:40 +02:00
2023-08-11 22:05:45 +10:00
update_now( ) {
# Update to newest version. First there should be an update ready, then we
# do the update, and then there should not be any ready anymore
2024-03-27 20:13:24 +01:00
" $SYSUPDATE " --definitions= " $WORKDIR /defs " --verify= no check-new
" $SYSUPDATE " --definitions= " $WORKDIR /defs " --verify= no update
( ! " $SYSUPDATE " --definitions= " $WORKDIR /defs " --verify= no check-new)
2023-08-11 22:05:45 +10:00
}
verify_version( ) {
2024-03-27 20:13:24 +01:00
local block_device = " ${ 1 : ? } "
local sector_size = " ${ 2 : ? } "
local version = " ${ 3 : ? } "
local part1_number = " ${ 4 : ? } "
local part2_number = " ${ 5 : ? } "
local gpt_reserved_sectors part1_offset part2_offset
2023-08-11 22:05:45 +10:00
2024-03-27 20:13:24 +01:00
gpt_reserved_sectors = $(( 1024 * 1024 / sector_size))
part1_offset = $(( ( part1_number - 1 ) * 2048 + gpt_reserved_sectors))
part2_offset = $(( ( part2_number - 1 ) * 2048 + gpt_reserved_sectors))
2023-08-11 22:05:45 +10:00
# Check the partitions
2024-03-27 20:13:24 +01:00
dd if = " $block_device " bs = " $sector_size " skip = " $part1_offset " count = 2048 | cmp " $WORKDIR /source/part1- $version .raw "
dd if = " $block_device " bs = " $sector_size " skip = " $part2_offset " count = 2048 | cmp " $WORKDIR /source/part2- $version .raw "
2023-08-11 22:05:45 +10:00
# Check the UKI
2024-03-27 20:13:24 +01:00
cmp " $WORKDIR /source/uki- $version .efi " " $WORKDIR /xbootldr/EFI/Linux/uki_ $version +3-0.efi "
test -z " $( ls -A " $WORKDIR /esp/EFI/Linux " ) "
2023-08-11 22:05:45 +10:00
2023-09-24 14:35:59 +02:00
# Check the extra efi
2024-03-27 20:13:24 +01:00
cmp " $WORKDIR /source/uki-extra- $version .efi " " $WORKDIR /xbootldr/EFI/Linux/uki_ $version .efi.extra.d/extra.addon.efi "
2023-09-24 14:35:59 +02:00
2023-08-11 22:05:45 +10:00
# Check the directories
2024-03-27 20:13:24 +01:00
cmp " $WORKDIR /source/dir- $version /foo.txt " " $WORKDIR /dirs/current/foo.txt "
cmp " $WORKDIR /source/dir- $version /bar.txt " " $WORKDIR /dirs/current/bar.txt "
2023-08-11 22:05:45 +10:00
}
2024-03-27 20:13:24 +01:00
for sector_size in " ${ SECTOR_SIZES [@] } " ; do
2023-08-11 22:05:45 +10:00
# Disk size of:
# - 1MB for GPT
# - 4 partitions of 2048 sectors each
# - 1MB for backup GPT
2024-03-27 20:13:24 +01:00
disk_size = $(( sector_size * 2048 * 4 + 1024 * 1024 * 2 ))
2023-08-11 22:05:45 +10:00
rm -f " $BACKING_FILE "
truncate -s " $disk_size " " $BACKING_FILE "
2024-03-27 19:35:30 +01:00
if [ [ -e /dev/loop-control ] ] ; then
2024-03-27 20:13:24 +01:00
blockdev = " $( losetup --find --show --sector-size " $sector_size " " $BACKING_FILE " ) "
2023-08-11 22:05:45 +10:00
else
blockdev = " $BACKING_FILE "
fi
sfdisk " $blockdev " <<EOF
2021-08-19 16:54:40 +02:00
label: gpt
unit: sectors
2023-08-11 22:05:45 +10:00
sector-size: $sector_size
2021-08-19 16:54:40 +02:00
size = 2048, type = 4f68bce3-e8cd-4db1-96e7-fbcaf984b709, name = _empty
size = 2048, type = 4f68bce3-e8cd-4db1-96e7-fbcaf984b709, name = _empty
size = 2048, type = 2c7357ed-ebd2-46d9-aec1-23d437ec2bf5, name = _empty
size = 2048, type = 2c7357ed-ebd2-46d9-aec1-23d437ec2bf5, name = _empty
EOF
2024-03-27 20:13:24 +01:00
for d in "dirs" "defs" ; do
rm -rf " ${ WORKDIR : ? } / $d "
mkdir -p " $WORKDIR / $d "
done
2021-08-19 16:54:40 +02:00
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/01-first.conf " <<EOF
2021-08-19 16:54:40 +02:00
[ Source]
Type = regular-file
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /source
2021-08-19 16:54:40 +02:00
MatchPattern = part1-@v.raw
[ Target]
Type = partition
2023-08-11 22:05:45 +10:00
Path = $blockdev
2021-08-19 16:54:40 +02:00
MatchPattern = part1-@v
MatchPartitionType = root-x86-64
EOF
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/02-second.conf " <<EOF
2021-08-19 16:54:40 +02:00
[ Source]
Type = regular-file
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /source
2021-08-19 16:54:40 +02:00
MatchPattern = part2-@v.raw.gz
[ Target]
Type = partition
2023-08-11 22:05:45 +10:00
Path = $blockdev
2021-08-19 16:54:40 +02:00
MatchPattern = part2-@v
MatchPartitionType = root-x86-64-verity
EOF
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/03-third.conf " <<EOF
2021-08-19 16:54:40 +02:00
[ Source]
Type = directory
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /source
2021-08-19 16:54:40 +02:00
MatchPattern = dir-@v
[ Target]
Type = directory
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /dirs
CurrentSymlink = $WORKDIR /dirs/current
2021-08-19 16:54:40 +02:00
MatchPattern = dir-@v
InstancesMax = 3
EOF
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/04-fourth.conf " <<EOF
2023-05-26 00:47:47 -04:00
[ Source]
Type = regular-file
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /source
2023-05-26 00:47:47 -04:00
MatchPattern = uki-@v.efi
[ Target]
Type = regular-file
Path = /EFI/Linux
PathRelativeTo = boot
MatchPattern = uki_@v+@l-@d.efi \
2023-08-11 22:05:45 +10:00
uki_@v+@l.efi \
uki_@v.efi
2023-05-26 00:47:47 -04:00
Mode = 0444
TriesLeft = 3
TriesDone = 0
InstancesMax = 2
2023-09-24 14:35:59 +02:00
EOF
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/05-fifth.conf " <<EOF
2023-09-24 14:35:59 +02:00
[ Source]
Type = regular-file
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /source
2023-09-24 14:35:59 +02:00
MatchPattern = uki-extra-@v.efi
[ Target]
Type = regular-file
Path = /EFI/Linux
PathRelativeTo = boot
MatchPattern = uki_@v.efi.extra.d/extra.addon.efi
Mode = 0444
InstancesMax = 2
2023-05-26 00:47:47 -04:00
EOF
2024-03-27 20:13:24 +01:00
rm -rf " ${ WORKDIR : ? } " /{ esp,xbootldr,source}
mkdir -p " $WORKDIR " /{ source,esp/EFI/Linux,xbootldr/EFI/Linux}
2023-08-11 22:05:45 +10:00
# Install initial version and verify
new_version " $sector_size " v1
update_now
verify_version " $blockdev " " $sector_size " v1 1 3
# Create second version, update and verify that it is added
new_version " $sector_size " v2
update_now
verify_version " $blockdev " " $sector_size " v2 2 4
# Create third version, update and verify it replaced the first version
new_version " $sector_size " v3
update_now
verify_version " $blockdev " " $sector_size " v3 1 3
2024-03-27 20:13:24 +01:00
test ! -f " $WORKDIR /xbootldr/EFI/Linux/uki_v1+3-0.efi "
test ! -f " $WORKDIR /xbootldr/EFI/Linux/uki_v1.efi.extra.d/extra.addon.efi "
test ! -d " $WORKDIR /xbootldr/EFI/Linux/uki_v1.efi.extra.d "
2023-08-11 22:05:45 +10:00
# Create fourth version, and update through a file:// URL. This should be
# almost as good as testing HTTP, but is simpler for us to set up. file:// is
# abstracted in curl for us, and since our main goal is to test our own code
# (and not curl) this test should be quite good even if not comprehensive. This
# will test the SHA256SUMS logic at least (we turn off GPG validation though,
# see above)
new_version " $sector_size " v4
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/02-second.conf " <<EOF
2021-08-19 16:54:40 +02:00
[ Source]
Type = url-file
2024-03-27 20:13:24 +01:00
Path = file://$WORKDIR /source
2021-08-19 16:54:40 +02:00
MatchPattern = part2-@v.raw.gz
[ Target]
Type = partition
2023-08-11 22:05:45 +10:00
Path = $blockdev
2021-08-19 16:54:40 +02:00
MatchPattern = part2-@v
MatchPartitionType = root-x86-64-verity
EOF
2024-03-27 20:13:24 +01:00
cat >" $WORKDIR /defs/03-third.conf " <<EOF
2021-08-19 16:54:40 +02:00
[ Source]
Type = url-tar
2024-03-27 20:13:24 +01:00
Path = file://$WORKDIR /source
2021-08-19 16:54:40 +02:00
MatchPattern = dir-@v.tar.gz
[ Target]
Type = directory
2024-03-27 20:13:24 +01:00
Path = $WORKDIR /dirs
CurrentSymlink = $WORKDIR /dirs/current
2021-08-19 16:54:40 +02:00
MatchPattern = dir-@v
InstancesMax = 3
EOF
2023-08-11 22:05:45 +10:00
update_now
verify_version " $blockdev " " $sector_size " v4 2 4
# Cleanup
2024-03-27 20:13:24 +01:00
[ [ -b " $blockdev " ] ] && losetup --detach " $blockdev "
2023-08-11 22:05:45 +10:00
rm " $BACKING_FILE "
done
2021-08-19 16:54:40 +02:00
2023-07-12 15:49:55 +02:00
touch /testok