#!/bin/bash -e # shellcheck disable=SC2119 run_sub_stage() { log "Begin ${SUB_STAGE_DIR}" pushd "${SUB_STAGE_DIR}" >/dev/null for i in {00..99}; do for DEBCONF_FILE in "${i}-debconf-${ARCH}" "${i}-debconf-${PLATFORM}-only" "${i}-debconf-arm-only" "${i}-debconf"; do if [ -f "${SUB_STAGE_DIR}/${DEBCONF_FILE}" ]; then # Skip arm-only if not arm if [[ "${DEBCONF_FILE}" == *"-debconf-arm-only" ]] && [[ "${ARCH}" != "armhf" && "${ARCH}" != "arm64" ]]; then continue fi log "Begin ${SUB_STAGE_DIR}/${DEBCONF_FILE}" on_chroot </dev/null if [ "${CLEAN}" = "1" ]; then rm -rf .pc rm -rf ./*-pc fi QUILT_PATCHES="${SUB_STAGE_DIR}/${PATCH_DIR}" SUB_STAGE_QUILT_PATCH_DIR="$(basename "$SUB_STAGE_DIR")-pc" mkdir -p "$SUB_STAGE_QUILT_PATCH_DIR" ln -snf "$SUB_STAGE_QUILT_PATCH_DIR" .pc quilt upgrade if [ -e "${SUB_STAGE_DIR}/${PATCH_DIR}/EDIT" ]; then echo "Dropping into bash to edit patches..." bash fi RC=0 quilt push -a || RC=$? case "$RC" in 0 | 2) ;; # 0 = success, 2 = "already applied" *) false ;; esac popd >/dev/null log "End ${SUB_STAGE_DIR}/${PATCH_DIR}" fi done for RUN_SCRIPT in "${i}-run-${ARCH}.sh" "${i}-run-${PLATFORM}-only.sh" "${i}-run-arm-only.sh" "${i}-run.sh"; do if [ -x "${SUB_STAGE_DIR}/${RUN_SCRIPT}" ]; then # Skip arm-only if not arm if [[ "${RUN_SCRIPT}" == *"-arm-only.sh" ]] && [[ "${ARCH}" != "armhf" && "${ARCH}" != "arm64" ]]; then continue fi log "Begin ${SUB_STAGE_DIR}/${RUN_SCRIPT}" "./${RUN_SCRIPT}" log "End ${SUB_STAGE_DIR}/${RUN_SCRIPT}" elif [ "${RUN_SCRIPT}" = "${i}-run.sh" ] && [ -f "${SUB_STAGE_DIR}/${i}-run.sh" ]; then log "Skip ${SUB_STAGE_DIR}/${i}-run.sh (not executable)" fi done for CHROOT_SCRIPT in "${i}-run-chroot-${ARCH}.sh" "${i}-run-chroot-${PLATFORM}-only.sh" "${i}-run-chroot-arm-only.sh" "${i}-run-chroot.sh"; do if [ -f "${SUB_STAGE_DIR}/${CHROOT_SCRIPT}" ]; then # Skip arm-only if not arm if [[ "${CHROOT_SCRIPT}" == *"-arm-only.sh" ]] && [[ "${ARCH}" != "armhf" && "${ARCH}" != "arm64" ]]; then continue fi log "Begin ${SUB_STAGE_DIR}/${CHROOT_SCRIPT}" on_chroot <"${SUB_STAGE_DIR}/${CHROOT_SCRIPT}" log "End ${SUB_STAGE_DIR}/${CHROOT_SCRIPT}" fi done done popd >/dev/null log "End ${SUB_STAGE_DIR}" } run_stage() { log "Begin ${STAGE_DIR}" STAGE="$(basename "${STAGE_DIR}")" pushd "${STAGE_DIR}" >/dev/null STAGE_WORK_DIR="${WORK_DIR}/${STAGE}" ROOTFS_DIR="${STAGE_WORK_DIR}"/rootfs unmount "${WORK_DIR}/${STAGE}" if [ ! -f SKIP ]; then if [ "${CLEAN}" = "1" ]; then if [ -d "${ROOTFS_DIR}" ]; then rm -rf "${ROOTFS_DIR}" fi fi if [ -x prerun.sh ]; then log "Begin ${STAGE_DIR}/prerun.sh" ./prerun.sh log "End ${STAGE_DIR}/prerun.sh" fi for SUB_STAGE_DIR in "${STAGE_DIR}"/*; do if [ -d "${SUB_STAGE_DIR}" ]; then if [ ! -f "${SUB_STAGE_DIR}/SKIP" ] && [ ! -f "${SUB_STAGE_DIR}/SKIP_${ARCH}" ] && { [ -z "${PLATFORM}" ] || [ ! -f "${SUB_STAGE_DIR}/SKIP_${PLATFORM}" ]; }; then run_sub_stage fi fi done fi unmount "${WORK_DIR}/${STAGE}" PREV_STAGE="${STAGE}" PREV_STAGE_DIR="${STAGE_DIR}" PREV_ROOTFS_DIR="${ROOTFS_DIR}" popd >/dev/null log "End ${STAGE_DIR}" } term() { if [ "$?" -ne 0 ]; then log "Build failed" else log "Build finished" fi unmount "${STAGE_WORK_DIR}" if [ "$STAGE" = "exports/img" ]; then for img in "${STAGE_WORK_DIR}/"*.img; do unmount_image "$img" done fi } if [ "$(id -u)" != "0" ]; then echo "Please run as root" 1>&2 exit 1 fi BASE_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)" if [[ $BASE_DIR = *" "* ]]; then echo "There is a space in the base path of pi-gen" echo "This is not a valid setup supported by debootstrap." echo "Please remove the spaces, or move pi-gen directory to a base path without spaces" 1>&2 exit 1 fi export BASE_DIR if [ -f config ]; then # shellcheck disable=SC1091 source config fi while getopts "c:" flag; do case "$flag" in c) EXTRA_CONFIG="$OPTARG" # shellcheck disable=SC1090 source "$EXTRA_CONFIG" ;; *) ;; esac done export PI_GEN=${PI_GEN:-pi-gen} export PI_GEN_REPO=${PI_GEN_REPO:-https://github.com/RPi-Distro/pi-gen} export PI_GEN_RELEASE=${PI_GEN_RELEASE:-Raspberry Pi reference} export ARCH="${ARCH:-arm64}" export PLATFORM="${PLATFORM:-}" export RELEASE=${RELEASE:-trixie} export IMG_NAME="${IMG_NAME:-vesperos-$RELEASE-$ARCH}" export USE_QEMU="${USE_QEMU:-0}" export IMG_DATE="${IMG_DATE:-"$(date +%Y-%m-%d)"}" export IMG_FILENAME="${IMG_FILENAME:-"${IMG_DATE}-${IMG_NAME}"}" export ARCHIVE_FILENAME="${ARCHIVE_FILENAME:-"image_${IMG_DATE}-${IMG_NAME}"}" export SCRIPT_DIR="${BASE_DIR}/scripts" export WORK_DIR="${WORK_DIR:-"${BASE_DIR}/work/${IMG_NAME}"}" export DEPLOY_DIR=${DEPLOY_DIR:-"${BASE_DIR}/deploy"} export DEPLOY_COMPRESSION=${DEPLOY_COMPRESSION:-xz} export COMPRESSION_LEVEL=${COMPRESSION_LEVEL:-6} export LOG_FILE="${WORK_DIR}/build.log" export TARGET_HOSTNAME=${TARGET_HOSTNAME:-vesperos} export WPA_COUNTRY export ENABLE_SSH="${ENABLE_SSH:-0}" export PUBKEY_ONLY_SSH="${PUBKEY_ONLY_SSH:-0}" export LOCALE_DEFAULT="${LOCALE_DEFAULT:-en_US.UTF-8}" export KEYBOARD_KEYMAP="${KEYBOARD_KEYMAP:-us}" export KEYBOARD_LAYOUT="${KEYBOARD_LAYOUT:-English (US)}" export TIMEZONE_DEFAULT="${TIMEZONE_DEFAULT:-America/Los_Angeles}" export GIT_HASH=${GIT_HASH:-"$(git rev-parse HEAD)"} export CLEAN export APT_PROXY export TEMP_REPO export STAGE export STAGE_DIR export STAGE_WORK_DIR export PREV_STAGE export PREV_STAGE_DIR export ROOTFS_DIR export PREV_ROOTFS_DIR export IMG_SUFFIX export EXPORT_DIR export EXPORT_ROOTFS_DIR export QUILT_PATCHES export QUILT_NO_DIFF_INDEX=1 export QUILT_NO_DIFF_TIMESTAMPS=1 export QUILT_REFRESH_ARGS="-p ab" export ENABLE_CLOUD_INIT=${ENABLE_CLOUD_INIT:-1} export LIVEBOOT=${LIVEBOOT:-0} export EXPORTS="${EXPORTS:-}" # shellcheck source=scripts/common source "${SCRIPT_DIR}/common" # shellcheck source=scripts/dependencies_check source "${SCRIPT_DIR}/dependencies_check" if [ "$SETFCAP" != "1" ]; then export CAPSH_ARG="--drop=cap_setfcap" fi mkdir -p "${WORK_DIR}" trap term EXIT INT TERM dependencies_check "${BASE_DIR}/depends" PAGESIZE=$(getconf PAGESIZE) if [ "$ARCH" == "armhf" ] && [ "$PAGESIZE" != "4096" ]; then echo echo "ERROR: Building an $ARCH image requires a kernel with a 4k page size (current: $PAGESIZE)" echo "On Raspberry Pi OS (64-bit), you can switch to a suitable kernel by adding the following to /boot/firmware/config.txt and rebooting:" echo echo "kernel=kernel8.img" echo "initramfs initramfs8 followkernel" echo exit 1 fi echo "Checking native $ARCH executable support..." if ! arch-test -n "$ARCH"; then echo "WARNING: Only a native build environment is supported. Checking emulated support..." if ! arch-test "$ARCH"; then echo "No fallback mechanism found. Ensure your OS has binfmt_misc support enabled and configured." exit 1 fi fi if [[ -n "${APT_PROXY}" ]] && ! curl --silent "${APT_PROXY}" >/dev/null; then echo "Could not reach APT_PROXY server: ${APT_PROXY}" exit 1 fi if [[ -n "${WPA_PASSWORD}" && ${#WPA_PASSWORD} -lt 8 || ${#WPA_PASSWORD} -gt 63 ]]; then echo "WPA_PASSWORD" must be between 8 and 63 characters exit 1 fi log "Begin ${BASE_DIR}" STAGE_LIST=${STAGE_LIST:-${BASE_DIR}/stage*} export STAGE_LIST for STAGE_DIR in $STAGE_LIST; do STAGE_DIR=$(realpath "${STAGE_DIR}") run_stage done CLEAN=1 # Exports declared in config: EXPORTS="stage3:squashfs stage2:img ..." for export_spec in ${EXPORTS}; do stage_name="${export_spec%%:*}" export_type="${export_spec##*:}" EXPORT_DIR="${BASE_DIR}/${stage_name}" EXPORT_ROOTFS_DIR="${WORK_DIR}/${stage_name}/rootfs" case "${export_type}" in img) STAGE_DIR="${BASE_DIR}/exports/img" run_stage ;; squashfs) STAGE_DIR="${BASE_DIR}/exports/squashfs" run_stage ;; noobs) if [ "${USE_QEMU}" != "1" ]; then STAGE_DIR="${BASE_DIR}/exports/noobs" run_stage fi ;; *) log "Unknown export type '${export_type}' in EXPORTS spec '${export_spec}'" false ;; esac done if [ -x "${BASE_DIR}/postrun.sh" ]; then log "Begin postrun.sh" cd "${BASE_DIR}" ./postrun.sh log "End postrun.sh" fi log "End ${BASE_DIR}"