2019-11-02 19:34:54 +03:00
|
|
|
#!/bin/sh
|
2019-06-04 15:16:30 +03:00
|
|
|
|
2022-01-31 14:42:52 +03:00
|
|
|
# Note: This is done before `set -e` to let `command` fail if needed
|
|
|
|
FUSE2FS_PATH=$(command -v fuse2fs)
|
2022-04-24 22:41:14 +03:00
|
|
|
RESIZE2FS_PATH=$(command -v resize2fs)
|
2022-01-31 14:42:52 +03:00
|
|
|
|
|
|
|
if [ -z "$FUSE2FS_PATH" ]; then
|
|
|
|
FUSE2FS_PATH=/usr/sbin/fuse2fs
|
|
|
|
fi
|
|
|
|
|
2022-04-24 22:41:14 +03:00
|
|
|
if [ -z "$RESIZE2FS_PATH" ]; then
|
|
|
|
RESIZE2FS_PATH=/usr/sbin/resize2fs
|
|
|
|
fi
|
|
|
|
|
2019-06-04 15:16:30 +03:00
|
|
|
set -e
|
|
|
|
|
2022-10-16 22:46:42 +03:00
|
|
|
SCRIPT_DIR="$(dirname "${0}")"
|
2022-06-30 21:48:51 +03:00
|
|
|
|
2022-10-16 22:46:42 +03:00
|
|
|
. "${SCRIPT_DIR}/.shell_include.sh"
|
2019-06-04 15:16:30 +03:00
|
|
|
|
2021-12-12 05:42:51 +03:00
|
|
|
USE_FUSE2FS=0
|
|
|
|
|
2019-11-02 19:34:54 +03:00
|
|
|
if [ "$(id -u)" != 0 ]; then
|
2022-01-31 14:42:52 +03:00
|
|
|
if [ -x "$FUSE2FS_PATH" ] && $FUSE2FS_PATH --help 2>&1 |grep fakeroot > /dev/null; then
|
2021-12-12 05:42:51 +03:00
|
|
|
USE_FUSE2FS=1
|
|
|
|
else
|
2022-09-03 13:54:05 +03:00
|
|
|
set +e
|
|
|
|
${SUDO} -E -- sh -c "\"$0\" $* || exit 42"
|
|
|
|
case $? in
|
|
|
|
1)
|
|
|
|
die "this script needs to run as root"
|
|
|
|
;;
|
|
|
|
42)
|
|
|
|
exit 1
|
|
|
|
;;
|
|
|
|
*)
|
|
|
|
exit 0
|
|
|
|
;;
|
|
|
|
esac
|
2021-12-12 05:42:51 +03:00
|
|
|
fi
|
2020-06-20 06:20:50 +03:00
|
|
|
else
|
|
|
|
: "${SUDO_UID:=0}" "${SUDO_GID:=0}"
|
2019-06-04 15:16:30 +03:00
|
|
|
fi
|
2020-06-20 04:01:02 +03:00
|
|
|
|
2020-01-02 05:06:14 +03:00
|
|
|
if [ "$(uname -s)" = "Darwin" ]; then
|
2019-12-25 22:47:43 +03:00
|
|
|
export PATH="/usr/local/opt/e2fsprogs/bin:$PATH"
|
|
|
|
export PATH="/usr/local/opt/e2fsprogs/sbin:$PATH"
|
2021-07-24 13:31:48 +03:00
|
|
|
export PATH="/opt/homebrew/opt/e2fsprogs/bin:$PATH"
|
|
|
|
export PATH="/opt/homebrew/opt/e2fsprogs/sbin:$PATH"
|
2022-03-30 10:34:47 +03:00
|
|
|
|
|
|
|
E2FSCK="e2fsck"
|
2022-10-27 14:50:07 +03:00
|
|
|
RESIZE2FS_PATH="resize2fs"
|
2022-11-06 02:56:52 +03:00
|
|
|
elif [ ! -f "$E2FSCK" ]; then
|
|
|
|
E2FSCK="$(command -v e2fsck)"
|
2022-03-30 10:34:47 +03:00
|
|
|
|
|
|
|
if [ ! -f "$E2FSCK" ]; then
|
2022-11-06 02:56:52 +03:00
|
|
|
E2FSCK="/usr/sbin/e2fsck"
|
|
|
|
if [ ! -f "$E2FSCK" ]; then
|
|
|
|
E2FSCK="/sbin/e2fsck"
|
|
|
|
fi
|
2022-03-30 10:34:47 +03:00
|
|
|
fi
|
2019-12-25 22:47:43 +03:00
|
|
|
fi
|
2020-07-28 19:20:06 +03:00
|
|
|
|
2021-07-03 11:38:56 +03:00
|
|
|
|
|
|
|
# Prepend the toolchain qemu directory so we pick up QEMU from there
|
|
|
|
PATH="$SCRIPT_DIR/../Toolchain/Local/qemu/bin:$PATH"
|
|
|
|
|
|
|
|
# Also prepend the i686 toolchain directory because that's where most
|
|
|
|
# people will have their QEMU binaries if they built them before the
|
|
|
|
# directory was changed to Toolchain/Local/qemu.
|
|
|
|
PATH="$SCRIPT_DIR/../Toolchain/Local/i686/bin:$PATH"
|
|
|
|
|
2021-08-17 19:27:55 +03:00
|
|
|
# We depend on GNU coreutils du for the --apparent-size extension.
|
|
|
|
# GNU coreutils is a build dependency.
|
2022-01-13 02:46:48 +03:00
|
|
|
if command -v gdu > /dev/null 2>&1 && gdu --version | grep -q "GNU coreutils"; then
|
2021-08-17 19:27:55 +03:00
|
|
|
GNUDU="gdu"
|
2021-05-17 00:50:46 +03:00
|
|
|
else
|
2021-08-17 19:27:55 +03:00
|
|
|
GNUDU="du"
|
2021-05-17 00:50:46 +03:00
|
|
|
fi
|
2021-08-17 19:27:55 +03:00
|
|
|
|
|
|
|
disk_usage() {
|
2022-05-23 23:44:36 +03:00
|
|
|
# shellcheck disable=SC2003,SC2307
|
2021-08-17 19:27:55 +03:00
|
|
|
expr "$(${GNUDU} -sk --apparent-size "$1" | cut -f1)"
|
2020-07-28 19:20:06 +03:00
|
|
|
}
|
|
|
|
|
2021-08-07 12:02:39 +03:00
|
|
|
inode_usage() {
|
|
|
|
find "$1" | wc -l
|
|
|
|
}
|
|
|
|
|
|
|
|
INODE_SIZE=128
|
|
|
|
INODE_COUNT=$(($(inode_usage "$SERENITY_SOURCE_DIR/Base") + $(inode_usage Root)))
|
2022-05-07 19:03:00 +03:00
|
|
|
INODE_COUNT=$((INODE_COUNT + 2000)) # Some additional inodes for toolchain files, could probably also be calculated
|
|
|
|
DISK_SIZE_BYTES=$((($(disk_usage "$SERENITY_SOURCE_DIR/Base") + $(disk_usage Root)) * 1024))
|
|
|
|
DISK_SIZE_BYTES=$((DISK_SIZE_BYTES + (INODE_COUNT * INODE_SIZE)))
|
2021-08-07 12:02:39 +03:00
|
|
|
|
2022-05-07 18:56:11 +03:00
|
|
|
if [ -z "$SERENITY_DISK_SIZE_BYTES" ]; then
|
|
|
|
# Try to use heuristics to guess a good disk size and inode count.
|
|
|
|
# The disk must notably fit:
|
|
|
|
# * Data blocks (for both files and directories),
|
|
|
|
# * Indirect/doubly indirect/triply indirect blocks,
|
|
|
|
# * Inodes and block bitmaps for each block group,
|
|
|
|
# * Plenty of extra free space and free inodes.
|
2022-05-07 19:03:00 +03:00
|
|
|
DISK_SIZE_BYTES=$((DISK_SIZE_BYTES * 2))
|
2022-05-07 18:56:11 +03:00
|
|
|
INODE_COUNT=$((INODE_COUNT * 7))
|
|
|
|
else
|
|
|
|
if [ "$DISK_SIZE_BYTES" -gt "$SERENITY_DISK_SIZE_BYTES" ]; then
|
|
|
|
die "SERENITY_DISK_SIZE_BYTES is set to $SERENITY_DISK_SIZE_BYTES but required disk size is $DISK_SIZE_BYTES bytes"
|
|
|
|
fi
|
|
|
|
DISK_SIZE_BYTES="$SERENITY_DISK_SIZE_BYTES"
|
|
|
|
fi
|
2020-07-28 19:20:06 +03:00
|
|
|
|
2021-04-23 21:55:57 +03:00
|
|
|
USE_EXISTING=0
|
2019-06-04 15:16:30 +03:00
|
|
|
|
2021-04-23 21:55:57 +03:00
|
|
|
if [ -f _disk_image ]; then
|
|
|
|
USE_EXISTING=1
|
|
|
|
|
|
|
|
echo "checking existing image"
|
|
|
|
result=0
|
2022-02-04 17:28:00 +03:00
|
|
|
"$E2FSCK" -f -y _disk_image || result=$?
|
2021-04-23 21:55:57 +03:00
|
|
|
if [ $result -ge 4 ]; then
|
|
|
|
rm -f _disk_image
|
|
|
|
USE_EXISTING=0
|
|
|
|
echo "failed, not using existing image"
|
2020-05-08 23:50:45 +03:00
|
|
|
else
|
2021-04-23 21:55:57 +03:00
|
|
|
echo "done"
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ $USE_EXISTING -eq 1 ]; then
|
|
|
|
OLD_DISK_SIZE_BYTES=$(wc -c < _disk_image)
|
2022-05-07 19:13:01 +03:00
|
|
|
if [ "$DISK_SIZE_BYTES" -gt "$OLD_DISK_SIZE_BYTES" ]; then
|
2021-04-23 21:55:57 +03:00
|
|
|
echo "resizing disk image..."
|
2022-05-07 19:13:01 +03:00
|
|
|
qemu-img resize -f raw _disk_image "$DISK_SIZE_BYTES" || die "could not resize disk image"
|
2022-04-24 22:41:14 +03:00
|
|
|
if ! "$RESIZE2FS_PATH" _disk_image; then
|
2021-04-23 21:55:57 +03:00
|
|
|
rm -f _disk_image
|
|
|
|
USE_EXISTING=0
|
|
|
|
echo "failed, not using existing image"
|
|
|
|
fi
|
|
|
|
echo "done"
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ $USE_EXISTING -ne 1 ]; then
|
|
|
|
printf "setting up disk image... "
|
2022-05-07 19:13:01 +03:00
|
|
|
qemu-img create -q -f raw _disk_image "$DISK_SIZE_BYTES" || die "could not create disk image"
|
2021-04-23 21:55:57 +03:00
|
|
|
chown "$SUDO_UID":"$SUDO_GID" _disk_image || die "could not adjust permissions on disk image"
|
|
|
|
echo "done"
|
|
|
|
|
|
|
|
printf "creating new filesystem... "
|
|
|
|
if [ "$(uname -s)" = "OpenBSD" ]; then
|
|
|
|
VND=$(vnconfig _disk_image)
|
|
|
|
(echo "e 0"; echo 83; echo n; echo 0; echo "*"; echo "quit") | fdisk -e "$VND"
|
2021-08-07 12:02:39 +03:00
|
|
|
newfs_ext2fs -D $INODE_SIZE -n $INODE_COUNT "/dev/r${VND}i" || die "could not create filesystem"
|
2021-04-23 21:55:57 +03:00
|
|
|
else
|
|
|
|
if [ -x /sbin/mke2fs ]; then
|
2021-08-07 12:02:39 +03:00
|
|
|
/sbin/mke2fs -q -I $INODE_SIZE -N $INODE_COUNT _disk_image || die "could not create filesystem"
|
2021-04-23 21:55:57 +03:00
|
|
|
else
|
2021-08-07 12:02:39 +03:00
|
|
|
mke2fs -q -I $INODE_SIZE -N $INODE_COUNT _disk_image || die "could not create filesystem"
|
2021-04-23 21:55:57 +03:00
|
|
|
fi
|
2020-05-07 01:03:32 +03:00
|
|
|
fi
|
2021-04-23 21:55:57 +03:00
|
|
|
echo "done"
|
2020-05-13 02:15:56 +03:00
|
|
|
fi
|
2020-05-08 23:50:45 +03:00
|
|
|
|
2020-05-13 02:15:56 +03:00
|
|
|
printf "mounting filesystem... "
|
|
|
|
mkdir -p mnt
|
|
|
|
use_genext2fs=0
|
2021-12-12 05:42:51 +03:00
|
|
|
if [ $USE_FUSE2FS -eq 1 ]; then
|
2022-01-31 14:42:52 +03:00
|
|
|
mount_cmd="$FUSE2FS_PATH _disk_image mnt/ -o fakeroot,rw"
|
2021-12-12 05:42:51 +03:00
|
|
|
elif [ "$(uname -s)" = "Darwin" ]; then
|
2021-12-12 05:28:16 +03:00
|
|
|
mount_cmd="fuse-ext2 _disk_image mnt -o rw+,allow_other,uid=501,gid=20"
|
2020-05-13 02:15:56 +03:00
|
|
|
elif [ "$(uname -s)" = "OpenBSD" ]; then
|
2021-12-12 05:28:16 +03:00
|
|
|
VND=$(vnconfig _disk_image)
|
|
|
|
mount_cmd="mount -t ext2fs "/dev/${VND}i" mnt/"
|
2020-05-13 02:15:56 +03:00
|
|
|
elif [ "$(uname -s)" = "FreeBSD" ]; then
|
2021-12-12 05:28:16 +03:00
|
|
|
MD=$(mdconfig _disk_image)
|
|
|
|
mount_cmd="fuse-ext2 -o rw+,direct_io "/dev/${MD}" mnt/"
|
2020-05-13 02:15:56 +03:00
|
|
|
else
|
2021-12-12 05:28:16 +03:00
|
|
|
mount_cmd="mount _disk_image mnt/"
|
2021-05-09 15:39:21 +03:00
|
|
|
fi
|
|
|
|
if ! eval "$mount_cmd"; then
|
2021-12-12 05:28:16 +03:00
|
|
|
if command -v genext2fs 1>/dev/null ; then
|
|
|
|
echo "mount failed but genext2fs exists, use it instead"
|
|
|
|
use_genext2fs=1
|
|
|
|
else
|
|
|
|
die "could not mount filesystem and genext2fs is missing"
|
|
|
|
fi
|
2021-05-09 15:39:21 +03:00
|
|
|
else
|
2021-12-12 05:28:16 +03:00
|
|
|
echo "done"
|
2019-12-25 22:47:43 +03:00
|
|
|
fi
|
2019-06-04 15:16:30 +03:00
|
|
|
|
|
|
|
cleanup() {
|
|
|
|
if [ -d mnt ]; then
|
2020-05-07 01:03:32 +03:00
|
|
|
if [ $use_genext2fs = 0 ] ; then
|
|
|
|
printf "unmounting filesystem... "
|
2021-12-12 05:42:51 +03:00
|
|
|
if [ $USE_FUSE2FS -eq 1 ]; then
|
|
|
|
fusermount -u mnt || (sleep 1 && sync && fusermount -u mnt)
|
|
|
|
else
|
|
|
|
umount mnt || ( sleep 1 && sync && umount mnt )
|
|
|
|
fi
|
2020-05-21 23:04:07 +03:00
|
|
|
rmdir mnt
|
|
|
|
else
|
|
|
|
rm -rf mnt
|
2020-05-07 01:03:32 +03:00
|
|
|
fi
|
2020-05-21 23:04:07 +03:00
|
|
|
|
2020-05-13 02:15:56 +03:00
|
|
|
if [ "$(uname -s)" = "OpenBSD" ]; then
|
|
|
|
vnconfig -u "$VND"
|
|
|
|
elif [ "$(uname -s)" = "FreeBSD" ]; then
|
|
|
|
mdconfig -d -u "$MD"
|
2020-01-02 05:06:14 +03:00
|
|
|
fi
|
2019-06-04 15:16:30 +03:00
|
|
|
echo "done"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
trap cleanup EXIT
|
|
|
|
|
2020-05-06 18:40:06 +03:00
|
|
|
script_path=$(cd -P -- "$(dirname -- "$0")" && pwd -P)
|
|
|
|
"$script_path/build-root-filesystem.sh"
|
2020-05-07 01:03:32 +03:00
|
|
|
|
|
|
|
if [ $use_genext2fs = 1 ]; then
|
|
|
|
# regenerate new image, since genext2fs is unable to reuse the previously written image.
|
|
|
|
# genext2fs is very slow in generating big images, so I use a smaller image here. size can be updated
|
|
|
|
# if it's not enough.
|
2021-08-07 12:02:39 +03:00
|
|
|
# not using "-I $INODE_SIZE" since it hangs. Serenity handles whatever default this uses instead.
|
2022-02-19 12:38:16 +03:00
|
|
|
genext2fs -B 4096 -b $((DISK_SIZE_BYTES / 4096)) -N $INODE_COUNT -d mnt _disk_image || die "try increasing image size (genext2fs -b)"
|
2020-05-07 01:03:32 +03:00
|
|
|
# if using docker with shared mount, file is created as root, so make it writable for users
|
|
|
|
chmod 0666 _disk_image
|
|
|
|
fi
|