#!/bin/sh # vim: set sw=4 ts=4 sts=4 et : SUBVOLUMES=' /var /var/log /var/tmp /etc ' cleanup() { if [ -n "${tmpdir}" ] && [ "${tmpdir}" != / ]; then if mountpoint -q "${tmpdir}"; then umount "${tmpdir}" fi rm -rf "${tmpdir}" unset tmpdir fi } copy_var() { dev="$1" echo 'Copying /var contents to data volume' mount -o subvol=var "${dev}" "${tmpdir}" || exit cp -acuv /var/. "${tmpdir}" || exit umount "${tmpdir}" } format_dev() { dev="$1" printf 'Creating BTRFS filesystem on %s\n' "${dev}" mkfs.btrfs "${dev}" || exit mount "${dev}" "${tmpdir}" || exit for vol in ${SUBVOLUMES}; do mkdir -p "${tmpdir}${vol%/*}" || exit btrfs subvolume create "${tmpdir}${vol}" || exit done relabel_all umount "${dev}" || exit } has_fs() { dev="$1" fstype=$(blkid -o value -s TYPE "${dev}") [ -n "${fstype}" ] } relabel_all() { selinuxtype=$(. /etc/selinux/config && echo ${SELINUXTYPE}) find "${tmpdir}" | \ setfiles \ -v \ -F \ -m \ -r "${tmpdir}" \ -s \ /etc/selinux/${selinuxtype}/contexts/files/file_contexts } setup_etc() { dev="$1" echo 'Initializing writable paths in /etc' mount -o subvol=etc "${dev}" "${tmpdir}" || exit if [ ! -f "${tmpdir}"/shadow ]; then cp -ca /etc/shadow "${tmpdir}"/shadow || exit fi mount -o bind "${tmpdir}"/shadow /etc/shadow || exit umount "${tmpdir}" } datapart=$(findfs PARTLABEL=dch-data) if [ -b "${datapart}" ]; then printf 'Found data partition: %s\n' "${datapart}" else echo 'Could not identify data partition' >&2 exit 1 fi trap cleanup INT TERM QUIT EXIT tmpdir=/run/storinit mkdir -p "${tmpdir}" if ! has_fs "${datapart}"; then format_dev "${datapart}" fi setup_etc "${datapart}" copy_var "${datapart}"