mirror of
https://github.com/Lakr233/vphone-cli.git
synced 2026-04-05 04:59:05 +08:00
1091 lines
31 KiB
Bash
Executable File
1091 lines
31 KiB
Bash
Executable File
#!/bin/zsh
|
|
# setup_machine.sh — Full vphone machine bootstrap through "First Boot".
|
|
#
|
|
# Runs README flow up to (but not including) "Subsequent Boots":
|
|
# 1) Host deps + project setup/build
|
|
# 2) vm_new + fw_prepare + fw_patch (or fw_patch_dev/ fw_patch_jb with --dev/--jb)
|
|
# 3) DFU restore (boot_dfu + restore_get_shsh + restore)
|
|
# 4) Ramdisk + CFW (boot_dfu + ramdisk_build + ramdisk_send + iproxy + cfw_install / cfw_install_dev / cfw_install_jb)
|
|
# 5) First boot launch (`make boot`) with printed in-guest commands
|
|
|
|
set -euo pipefail
|
|
|
|
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
|
|
PROJECT_ROOT="$(cd "${SCRIPT_DIR}/.." && pwd)"
|
|
|
|
cd "$PROJECT_ROOT"
|
|
|
|
LOG_DIR="${PROJECT_ROOT}/setup_logs"
|
|
DFU_LOG="${LOG_DIR}/boot_dfu.log"
|
|
IPROXY_LOG=""
|
|
BOOT_LOG="${LOG_DIR}/boot.log"
|
|
|
|
DFU_PID=""
|
|
IPROXY_PID=""
|
|
BOOT_PID=""
|
|
BOOT_FIFO=""
|
|
BOOT_FIFO_FD=""
|
|
SUDO_ASKPASS_SCRIPT=""
|
|
|
|
VM_DIR="${VM_DIR:-vm}"
|
|
VM_DIR_ABS="${VM_DIR:A}"
|
|
AUTO_KILL_VM_LOCKS="${AUTO_KILL_VM_LOCKS:-1}"
|
|
POST_RESTORE_KILL_DELAY="${POST_RESTORE_KILL_DELAY:-30}"
|
|
POST_KILL_SETTLE_DELAY="${POST_KILL_SETTLE_DELAY:-5}"
|
|
RAMDISK_SSH_TIMEOUT="${RAMDISK_SSH_TIMEOUT:-60}"
|
|
RAMDISK_SSH_INTERVAL="${RAMDISK_SSH_INTERVAL:-2}"
|
|
RAMDISK_SSH_PORT="${RAMDISK_SSH_PORT:-}"
|
|
RAMDISK_SSH_USER="${RAMDISK_SSH_USER:-root}"
|
|
RAMDISK_SSH_PASS="${RAMDISK_SSH_PASS:-alpine}"
|
|
IPROXY_UDID="${IPROXY_UDID:-}"
|
|
IPROXY_DEVICE_WAIT_TIMEOUT="${IPROXY_DEVICE_WAIT_TIMEOUT:-90}"
|
|
IPROXY_DEVICE_WAIT_INTERVAL="${IPROXY_DEVICE_WAIT_INTERVAL:-1}"
|
|
RAMDISK_SSH_PORT_EXPLICIT=0
|
|
if [[ -n "$RAMDISK_SSH_PORT" ]]; then
|
|
RAMDISK_SSH_PORT_EXPLICIT=1
|
|
fi
|
|
|
|
DEVICE_UDID=""
|
|
DEVICE_ECID=""
|
|
IPROXY_TARGET_UDID=""
|
|
IPROXY_RESOLVE_REASON=""
|
|
BOOT_ANALYSIS_TIMEOUT="${BOOT_ANALYSIS_TIMEOUT:-300}"
|
|
BOOT_PROMPT_FALLBACK_TIMEOUT="${BOOT_PROMPT_FALLBACK_TIMEOUT:-60}"
|
|
BOOT_BASH_PROMPT_REGEX="${BOOT_BASH_PROMPT_REGEX:-bash-[0-9]+(\.[0-9]+)+#}"
|
|
BOOT_PANIC_REGEX="${BOOT_PANIC_REGEX:-panic|kernel panic|panic\\.apple\\.com|stackshot succeeded}"
|
|
PMD3_BRIDGE="${PMD3_BRIDGE:-${PROJECT_ROOT}/scripts/pymobiledevice3_bridge.py}"
|
|
NONE_INTERACTIVE_RAW="${NONE_INTERACTIVE:-0}"
|
|
NONE_INTERACTIVE=0
|
|
JB_MODE=0
|
|
DEV_MODE=0
|
|
SKIP_PROJECT_SETUP=0
|
|
|
|
die() {
|
|
echo "[-] $*" >&2
|
|
exit 1
|
|
}
|
|
|
|
require_cmd() {
|
|
local cmd="$1"
|
|
command -v "$cmd" >/dev/null 2>&1 || die "Missing required command: $cmd"
|
|
}
|
|
|
|
find_python_for_pmd3() {
|
|
local candidate
|
|
for candidate in \
|
|
"${PROJECT_ROOT}/.venv/bin/python3" \
|
|
"$(command -v python3 2>/dev/null || true)"
|
|
do
|
|
[[ -n "$candidate" ]] || continue
|
|
[[ -x "$candidate" ]] || continue
|
|
if "$candidate" -c "import pymobiledevice3" >/dev/null 2>&1; then
|
|
echo "$candidate"
|
|
return 0
|
|
fi
|
|
done
|
|
return 1
|
|
}
|
|
|
|
normalize_ecid() {
|
|
local ecid="$1"
|
|
ecid="${ecid#0x}"
|
|
ecid="${ecid#0X}"
|
|
[[ "$ecid" =~ ^[0-9A-Fa-f]{1,16}$ ]] || return 1
|
|
printf "%016s" "${ecid:u}" | tr ' ' '0'
|
|
}
|
|
|
|
load_device_identity() {
|
|
local prediction_file="${VM_DIR_ABS}/udid-prediction.txt"
|
|
local timeout=30
|
|
local waited=0
|
|
local key value
|
|
local udid_ecid
|
|
|
|
while [[ ! -f "$prediction_file" && "$waited" -lt "$timeout" ]]; do
|
|
if [[ -n "$DFU_PID" ]] && ! kill -0 "$DFU_PID" 2>/dev/null; then
|
|
break
|
|
fi
|
|
sleep 1
|
|
waited=$(( waited + 1 ))
|
|
done
|
|
|
|
[[ -f "$prediction_file" ]] || die "Missing ${prediction_file}. Rebuild and run make boot_dfu to generate it."
|
|
|
|
DEVICE_UDID=""
|
|
DEVICE_ECID=""
|
|
while IFS='=' read -r key value; do
|
|
case "$key" in
|
|
UDID)
|
|
DEVICE_UDID="${value:u}"
|
|
;;
|
|
ECID)
|
|
DEVICE_ECID="$(normalize_ecid "$value" || true)"
|
|
;;
|
|
esac
|
|
done < "$prediction_file"
|
|
|
|
[[ "$DEVICE_UDID" =~ ^[0-9A-F]{8}-[0-9A-F]{16}$ ]] \
|
|
|| die "Invalid UDID in ${prediction_file}: ${DEVICE_UDID}"
|
|
|
|
if [[ -z "$DEVICE_ECID" ]]; then
|
|
DEVICE_ECID="${DEVICE_UDID#*-}"
|
|
fi
|
|
[[ "$DEVICE_ECID" =~ ^[0-9A-F]{16}$ ]] \
|
|
|| die "Invalid ECID in ${prediction_file}: ${DEVICE_ECID}"
|
|
|
|
udid_ecid="${DEVICE_UDID#*-}"
|
|
[[ "$udid_ecid" == "$DEVICE_ECID" ]] \
|
|
|| die "UDID/ECID mismatch in ${prediction_file}: ${DEVICE_UDID} vs 0x${DEVICE_ECID}"
|
|
|
|
echo "[+] Device identity loaded: UDID=${DEVICE_UDID} ECID=0x${DEVICE_ECID}"
|
|
}
|
|
|
|
list_usbmux_udids() {
|
|
local pmd3_python
|
|
pmd3_python="$(find_python_for_pmd3 || true)"
|
|
[[ -x "$pmd3_python" ]] || die "pymobiledevice3 python runtime not found (run: make setup_tools)"
|
|
[[ -f "$PMD3_BRIDGE" ]] || die "Missing bridge script: $PMD3_BRIDGE"
|
|
"$pmd3_python" "$PMD3_BRIDGE" usbmux-list 2>/dev/null | tr -d '\r' | sed '/^[[:space:]]*$/d'
|
|
}
|
|
|
|
print_usbmux_udids() {
|
|
local -a udids
|
|
local udid
|
|
udids=(${(@f)$(list_usbmux_udids)})
|
|
|
|
if (( ${#udids[@]} == 0 )); then
|
|
echo " (none)"
|
|
return
|
|
fi
|
|
|
|
for udid in "${udids[@]}"; do
|
|
echo " - ${udid}"
|
|
done
|
|
}
|
|
|
|
try_resolve_iproxy_target_udid() {
|
|
local -a usbmux_udids ecid_matches
|
|
local udid
|
|
local ecid_lower
|
|
|
|
IPROXY_TARGET_UDID=""
|
|
IPROXY_RESOLVE_REASON=""
|
|
|
|
if [[ -n "$IPROXY_UDID" ]]; then
|
|
IPROXY_TARGET_UDID="$IPROXY_UDID"
|
|
IPROXY_RESOLVE_REASON="override"
|
|
return 0
|
|
fi
|
|
|
|
usbmux_udids=(${(@f)$(list_usbmux_udids)})
|
|
if (( ${#usbmux_udids[@]} == 0 )); then
|
|
IPROXY_RESOLVE_REASON="none"
|
|
return 1
|
|
fi
|
|
|
|
for udid in "${usbmux_udids[@]}"; do
|
|
if [[ "$udid" == "$DEVICE_UDID" ]]; then
|
|
IPROXY_TARGET_UDID="$udid"
|
|
IPROXY_RESOLVE_REASON="restore_match"
|
|
return 0
|
|
fi
|
|
done
|
|
|
|
ecid_lower="${DEVICE_ECID:l}"
|
|
ecid_matches=()
|
|
for udid in "${usbmux_udids[@]}"; do
|
|
if [[ "${udid:l}" == *"${ecid_lower}"* ]]; then
|
|
ecid_matches+=("$udid")
|
|
fi
|
|
done
|
|
|
|
if (( ${#ecid_matches[@]} == 1 )); then
|
|
IPROXY_TARGET_UDID="${ecid_matches[1]}"
|
|
IPROXY_RESOLVE_REASON="ecid_match"
|
|
return 0
|
|
fi
|
|
|
|
if (( ${#usbmux_udids[@]} == 1 )); then
|
|
IPROXY_RESOLVE_REASON="single_mismatch"
|
|
return 3
|
|
fi
|
|
|
|
IPROXY_RESOLVE_REASON="ambiguous"
|
|
return 2
|
|
}
|
|
|
|
wait_for_iproxy_target_udid() {
|
|
local timeout interval waited rc
|
|
|
|
timeout="$IPROXY_DEVICE_WAIT_TIMEOUT"
|
|
interval="$IPROXY_DEVICE_WAIT_INTERVAL"
|
|
waited=0
|
|
|
|
[[ "$timeout" == <-> ]] || die "IPROXY_DEVICE_WAIT_TIMEOUT must be an integer (seconds)"
|
|
[[ "$interval" == <-> ]] || die "IPROXY_DEVICE_WAIT_INTERVAL must be an integer (seconds)"
|
|
(( timeout > 0 )) || die "IPROXY_DEVICE_WAIT_TIMEOUT must be > 0"
|
|
(( interval > 0 )) || die "IPROXY_DEVICE_WAIT_INTERVAL must be > 0"
|
|
|
|
echo "[*] Resolving iproxy target UDID (timeout=${timeout}s)..."
|
|
while (( waited < timeout )); do
|
|
if try_resolve_iproxy_target_udid; then
|
|
echo "[*] USBMux IDs currently visible:"
|
|
print_usbmux_udids
|
|
case "$IPROXY_RESOLVE_REASON" in
|
|
override)
|
|
echo "[*] Using explicit IPROXY_UDID override: ${IPROXY_TARGET_UDID}"
|
|
;;
|
|
restore_match)
|
|
echo "[+] iproxy target UDID matched restore UDID: ${IPROXY_TARGET_UDID}"
|
|
;;
|
|
ecid_match)
|
|
echo "[+] iproxy target UDID matched ECID substring: ${IPROXY_TARGET_UDID}"
|
|
;;
|
|
esac
|
|
return
|
|
fi
|
|
rc=$?
|
|
|
|
if (( waited == 0 || waited % 5 == 0 )); then
|
|
case "$rc" in
|
|
2)
|
|
echo " waiting for USBMux disambiguation... ${waited}s elapsed"
|
|
;;
|
|
3)
|
|
echo " waiting for restore UDID/ECID match (strict mode)... ${waited}s elapsed"
|
|
;;
|
|
*)
|
|
echo " waiting for USBMux device... ${waited}s elapsed"
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
sleep "$interval"
|
|
(( waited += interval ))
|
|
done
|
|
|
|
echo "[-] Timed out resolving iproxy target UDID after ${timeout}s."
|
|
echo "[-] USBMux IDs currently visible:"
|
|
print_usbmux_udids
|
|
if [[ "$IPROXY_RESOLVE_REASON" == "single_mismatch" ]]; then
|
|
die "Only non-matching USBMux device was visible. Strict identity isolation is enabled; wait for restore UDID/ECID or set IPROXY_UDID explicitly."
|
|
fi
|
|
if [[ "$IPROXY_RESOLVE_REASON" == "ambiguous" ]]; then
|
|
die "Multiple USBMux devices detected and none uniquely matched restore UDID/ECID. Set IPROXY_UDID explicitly."
|
|
fi
|
|
die "No USBMux devices detected for iproxy. Ensure ramdisk has fully booted USB stack."
|
|
}
|
|
|
|
port_is_listening() {
|
|
local port="$1"
|
|
lsof -n -t -iTCP:"$port" -sTCP:LISTEN >/dev/null 2>&1
|
|
}
|
|
|
|
pick_random_ssh_port() {
|
|
local attempt port
|
|
for attempt in {1..200}; do
|
|
port=$((20000 + (RANDOM % 40000)))
|
|
if ! port_is_listening "$port"; then
|
|
echo "$port"
|
|
return 0
|
|
fi
|
|
done
|
|
return 1
|
|
}
|
|
|
|
choose_ramdisk_ssh_port() {
|
|
if [[ -n "$RAMDISK_SSH_PORT" ]]; then
|
|
[[ "$RAMDISK_SSH_PORT" == <-> ]] || die "RAMDISK_SSH_PORT must be an integer"
|
|
(( RAMDISK_SSH_PORT >= 1 && RAMDISK_SSH_PORT <= 65535 )) \
|
|
|| die "RAMDISK_SSH_PORT out of range: ${RAMDISK_SSH_PORT}"
|
|
if port_is_listening "$RAMDISK_SSH_PORT"; then
|
|
die "RAMDISK_SSH_PORT ${RAMDISK_SSH_PORT} is already in use"
|
|
fi
|
|
return
|
|
fi
|
|
|
|
RAMDISK_SSH_PORT="$(pick_random_ssh_port)" \
|
|
|| die "Failed to allocate a random local SSH forward port"
|
|
}
|
|
|
|
parse_bool() {
|
|
local raw="${1:-0}"
|
|
# zsh `${var:l}` lowercases value for tolerant bool parsing.
|
|
case "${raw:l}" in
|
|
1|true|yes|on) return 0 ;;
|
|
*) return 1 ;;
|
|
esac
|
|
}
|
|
|
|
setup_sudo_noninteractive() {
|
|
[[ -n "${SUDO_PASSWORD:-}" ]] || return 0
|
|
|
|
SUDO_ASKPASS_SCRIPT="$(mktemp "${TMPDIR:-/tmp}/vphone-sudo-askpass.XXXXXX")"
|
|
cat >"$SUDO_ASKPASS_SCRIPT" <<'EOF'
|
|
#!/bin/sh
|
|
printf '%s\n' "${SUDO_PASSWORD:-}"
|
|
EOF
|
|
chmod 700 "$SUDO_ASKPASS_SCRIPT"
|
|
export SUDO_ASKPASS="$SUDO_ASKPASS_SCRIPT"
|
|
|
|
if sudo -A -v >/dev/null 2>&1; then
|
|
echo "[+] sudo credential preloaded via SUDO_PASSWORD"
|
|
else
|
|
echo "[!] SUDO_PASSWORD provided but sudo -A validation failed; continuing without preload"
|
|
fi
|
|
}
|
|
|
|
collect_vm_lock_pids() {
|
|
local -a paths pids
|
|
local path pid
|
|
typeset -U pids
|
|
|
|
paths=(
|
|
"${VM_DIR_ABS}/nvram.bin"
|
|
"${VM_DIR_ABS}/machineIdentifier.bin"
|
|
"${VM_DIR_ABS}/Disk.img"
|
|
"${VM_DIR_ABS}/SEPStorage"
|
|
)
|
|
|
|
for path in "${paths[@]}"; do
|
|
[[ -e "$path" ]] || continue
|
|
while IFS= read -r pid; do
|
|
[[ "$pid" == <-> ]] || continue
|
|
[[ "$pid" == "$$" ]] && continue
|
|
pids+=("$pid")
|
|
done < <(lsof -t -- "$path" 2>/dev/null || true)
|
|
done
|
|
|
|
(( ${#pids[@]} > 0 )) && print -l -- "${pids[@]}" || true
|
|
}
|
|
|
|
check_vm_storage_locks() {
|
|
if ! command -v lsof >/dev/null 2>&1; then
|
|
echo "[!] lsof not found; skipping VM lock preflight."
|
|
return
|
|
fi
|
|
|
|
local -a lock_pids
|
|
lock_pids=(${(@f)$(collect_vm_lock_pids)})
|
|
(( ${#lock_pids[@]} == 0 )) && return
|
|
|
|
echo "[-] VM storage files are currently in use: ${VM_DIR_ABS}"
|
|
echo " This usually means another vphone process is still running."
|
|
|
|
local pid proc_info
|
|
for pid in "${lock_pids[@]}"; do
|
|
[[ -z "$pid" || "$pid" == "$$" ]] && continue
|
|
proc_info="$(ps -o pid=,ppid=,command= -p "$pid" 2>/dev/null || true)"
|
|
[[ -n "$proc_info" ]] && echo " $proc_info" || echo " pid=$pid"
|
|
done
|
|
|
|
if [[ "$AUTO_KILL_VM_LOCKS" == "1" ]]; then
|
|
echo "[*] AUTO_KILL_VM_LOCKS=1 set; terminating lock holder processes..."
|
|
for pid in "${lock_pids[@]}"; do
|
|
[[ -z "$pid" || "$pid" == "$$" ]] && continue
|
|
stop_process_tree "$pid"
|
|
done
|
|
sleep 1
|
|
|
|
lock_pids=(${(@f)$(collect_vm_lock_pids)})
|
|
(( ${#lock_pids[@]} == 0 )) && { echo "[+] Cleared VM storage locks"; return; }
|
|
echo "[-] VM storage locks still present after AUTO_KILL_VM_LOCKS attempt."
|
|
fi
|
|
|
|
die "Stop those processes and retry. You can also set AUTO_KILL_VM_LOCKS=1."
|
|
}
|
|
|
|
list_descendants() {
|
|
local pid
|
|
local -a children
|
|
|
|
children=("${(@f)$(pgrep -P "$1" 2>/dev/null || true)}")
|
|
for pid in "${children[@]}"; do
|
|
[[ -z "$pid" ]] && continue
|
|
list_descendants "$pid"
|
|
print -r -- "$pid"
|
|
done
|
|
}
|
|
|
|
kill_descendants() {
|
|
local -a descendants
|
|
descendants=("${(@f)$(list_descendants "$1")}")
|
|
[[ ${#descendants[@]} -gt 0 ]] && kill -9 "${descendants[@]}" >/dev/null 2>&1 || true
|
|
}
|
|
|
|
stop_process_tree() {
|
|
local pid="$1"
|
|
[[ -n "$pid" && "$pid" == <-> ]] || return 0
|
|
kill -0 "$pid" 2>/dev/null || return 0
|
|
|
|
kill_descendants "$pid"
|
|
kill "$pid" >/dev/null 2>&1 || true
|
|
sleep 1
|
|
if kill -0 "$pid" 2>/dev/null; then
|
|
kill -9 "$pid" >/dev/null 2>&1 || true
|
|
fi
|
|
wait "$pid" 2>/dev/null || true
|
|
}
|
|
|
|
kill_stale_vphone_procs() {
|
|
local vphone_bin="${PROJECT_ROOT}/.build/release/vphone-cli"
|
|
local -a stale_pids
|
|
stale_pids=(${(@f)$(pgrep -f "$vphone_bin" 2>/dev/null || true)})
|
|
(( ${#stale_pids[@]} == 0 )) && return
|
|
|
|
echo "[*] Found stale vphone-cli process(es) (pids: ${stale_pids[*]}); terminating..."
|
|
for pid in "${stale_pids[@]}"; do
|
|
[[ "$pid" == "$$" ]] && continue
|
|
stop_process_tree "$pid"
|
|
done
|
|
|
|
# Wait up to 8s for VZ file locks to clear (flock/fcntl locks may lag behind process exit)
|
|
local waited=0
|
|
while (( waited < 8 )); do
|
|
local -a remaining
|
|
remaining=(${(@f)$(collect_vm_lock_pids)})
|
|
(( ${#remaining[@]} == 0 )) && break
|
|
sleep 1
|
|
waited=$(( waited + 1 ))
|
|
done
|
|
echo "[+] Stale vphone-cli processes cleared"
|
|
}
|
|
|
|
force_release_vm_locks() {
|
|
local -a lock_pids
|
|
local pid
|
|
|
|
lock_pids=(${(@f)$(collect_vm_lock_pids)})
|
|
(( ${#lock_pids[@]} == 0 )) && return
|
|
|
|
echo "[*] Releasing lingering VM lock holders..."
|
|
for pid in "${lock_pids[@]}"; do
|
|
[[ -z "$pid" || "$pid" == "$$" ]] && continue
|
|
stop_process_tree "$pid"
|
|
done
|
|
|
|
sleep 1
|
|
}
|
|
|
|
cleanup() {
|
|
if [[ -n "$BOOT_FIFO_FD" ]]; then
|
|
exec {BOOT_FIFO_FD}>&- || true
|
|
BOOT_FIFO_FD=""
|
|
fi
|
|
|
|
if [[ -n "$BOOT_PID" ]] && kill -0 "$BOOT_PID" 2>/dev/null; then
|
|
stop_process_tree "$BOOT_PID"
|
|
BOOT_PID=""
|
|
fi
|
|
|
|
if [[ -n "$BOOT_FIFO" && -p "$BOOT_FIFO" ]]; then
|
|
rm -f "$BOOT_FIFO" || true
|
|
BOOT_FIFO=""
|
|
fi
|
|
|
|
if [[ -n "$IPROXY_PID" ]]; then
|
|
stop_process_tree "$IPROXY_PID"
|
|
IPROXY_PID=""
|
|
fi
|
|
|
|
if [[ -n "$DFU_PID" ]]; then
|
|
stop_process_tree "$DFU_PID"
|
|
DFU_PID=""
|
|
fi
|
|
|
|
if [[ -n "$SUDO_ASKPASS_SCRIPT" && -f "$SUDO_ASKPASS_SCRIPT" ]]; then
|
|
rm -f "$SUDO_ASKPASS_SCRIPT" || true
|
|
SUDO_ASKPASS_SCRIPT=""
|
|
fi
|
|
}
|
|
|
|
start_first_boot() {
|
|
check_vm_storage_locks
|
|
mkdir -p "$LOG_DIR"
|
|
: > "$BOOT_LOG"
|
|
|
|
BOOT_FIFO="$(mktemp -u "${TMPDIR:-/tmp}/vphone-first-boot.XXXXXX")"
|
|
mkfifo "$BOOT_FIFO"
|
|
|
|
(make boot <"$BOOT_FIFO" >"$BOOT_LOG" 2>&1) &
|
|
BOOT_PID=$!
|
|
|
|
exec {BOOT_FIFO_FD}>"$BOOT_FIFO"
|
|
|
|
sleep 2
|
|
if ! kill -0 "$BOOT_PID" 2>/dev/null; then
|
|
die "make boot exited early during first boot stage"
|
|
fi
|
|
}
|
|
|
|
send_first_boot_commands() {
|
|
[[ -n "$BOOT_FIFO_FD" ]] || die "First boot command channel is not open"
|
|
|
|
local commands=(
|
|
"export PATH='/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/bin/X11:/usr/games:/iosbinpack64/usr/local/sbin:/iosbinpack64/usr/local/bin:/iosbinpack64/usr/sbin:/iosbinpack64/usr/bin:/iosbinpack64/sbin:/iosbinpack64/bin'"
|
|
"cp /iosbinpack64/etc/profile /var/profile"
|
|
"cp /iosbinpack64/etc/motd /var/motd"
|
|
"mkdir -p /var/dropbear"
|
|
"dropbearkey -t rsa -f /var/dropbear/dropbear_rsa_host_key"
|
|
"dropbearkey -t ecdsa -f /var/dropbear/dropbear_ecdsa_host_key"
|
|
"shutdown -h now"
|
|
)
|
|
|
|
local cmd
|
|
for cmd in "${commands[@]}"; do
|
|
print -r -- "$cmd" >&${BOOT_FIFO_FD}
|
|
done
|
|
}
|
|
|
|
monitor_boot_log_until() {
|
|
local timeout="$1"
|
|
local waited=0
|
|
|
|
[[ "$timeout" == <-> ]] || die "monitor timeout must be integer seconds"
|
|
(( timeout > 0 )) || die "monitor timeout must be > 0"
|
|
|
|
while (( waited < timeout )); do
|
|
if [[ -f "$BOOT_LOG" ]] && grep -Eiq "$BOOT_PANIC_REGEX" "$BOOT_LOG"; then
|
|
echo "panic"
|
|
return 0
|
|
fi
|
|
if [[ -f "$BOOT_LOG" ]] && grep -Eq "$BOOT_BASH_PROMPT_REGEX" "$BOOT_LOG"; then
|
|
echo "bash"
|
|
return 0
|
|
fi
|
|
if [[ -n "$BOOT_PID" ]] && ! kill -0 "$BOOT_PID" 2>/dev/null; then
|
|
echo "exited"
|
|
return 0
|
|
fi
|
|
sleep 1
|
|
waited=$(( waited + 1 ))
|
|
done
|
|
|
|
echo "timeout"
|
|
}
|
|
|
|
wait_for_first_boot_prompt_auto() {
|
|
local boot_state
|
|
boot_state="$(monitor_boot_log_until "$BOOT_PROMPT_FALLBACK_TIMEOUT")"
|
|
case "$boot_state" in
|
|
panic)
|
|
echo "[-] Panic detected while waiting for first-boot shell prompt."
|
|
tail -n 80 "$BOOT_LOG" 2>/dev/null || true
|
|
die "First boot panicked before command injection."
|
|
;;
|
|
bash)
|
|
echo "[+] First-boot shell prompt detected"
|
|
;;
|
|
exited)
|
|
echo "[-] make boot exited before first-boot command injection."
|
|
tail -n 80 "$BOOT_LOG" 2>/dev/null || true
|
|
die "First boot exited before command injection."
|
|
;;
|
|
timeout)
|
|
echo "[!] Shell prompt not detected within ${BOOT_PROMPT_FALLBACK_TIMEOUT}s; fallback to timed continue."
|
|
;;
|
|
esac
|
|
}
|
|
|
|
wait_for_device_ssh() {
|
|
local port="${1:-22222}"
|
|
local timeout="${2:-120}"
|
|
local pass="${3:-alpine}"
|
|
local sshpass_bin waited=0
|
|
|
|
sshpass_bin="$(command -v sshpass || true)"
|
|
[[ -x "$sshpass_bin" ]] || die "sshpass not found (run: make setup_tools)"
|
|
|
|
echo "[*] Waiting for device SSH on localhost:${port} (timeout=${timeout}s)..."
|
|
while (( waited < timeout )); do
|
|
if [[ -n "$BOOT_PID" ]] && ! kill -0 "$BOOT_PID" 2>/dev/null; then
|
|
die "VM exited while waiting for device SSH."
|
|
fi
|
|
if "$sshpass_bin" -p "$pass" ssh \
|
|
-o StrictHostKeyChecking=no \
|
|
-o UserKnownHostsFile=/dev/null \
|
|
-o PreferredAuthentications=password \
|
|
-o ConnectTimeout=5 -q \
|
|
-p "$port" root@localhost "echo ready" >/dev/null 2>&1; then
|
|
echo "[+] Device SSH is ready on port ${port}"
|
|
return
|
|
fi
|
|
if (( waited == 0 || waited % 10 == 0 )); then
|
|
echo " waiting... ${waited}s elapsed"
|
|
fi
|
|
sleep 2
|
|
(( waited += 2 ))
|
|
done
|
|
die "Device SSH not ready after ${timeout}s"
|
|
}
|
|
|
|
halt_device_ssh() {
|
|
local port="${1:-22222}"
|
|
local pass="${2:-alpine}"
|
|
local sshpass_bin
|
|
sshpass_bin="$(command -v sshpass)"
|
|
echo "[*] Halting device via SSH..."
|
|
"$sshpass_bin" -p "$pass" ssh \
|
|
-o StrictHostKeyChecking=no \
|
|
-o UserKnownHostsFile=/dev/null \
|
|
-o PreferredAuthentications=password \
|
|
-o ConnectTimeout=10 -q \
|
|
-p "$port" root@localhost "halt" 2>/dev/null || true
|
|
}
|
|
|
|
run_boot_analysis() {
|
|
local boot_state
|
|
|
|
check_vm_storage_locks
|
|
mkdir -p "$LOG_DIR"
|
|
: > "$BOOT_LOG"
|
|
(make boot >"$BOOT_LOG" 2>&1) &
|
|
BOOT_PID=$!
|
|
|
|
sleep 2
|
|
if ! kill -0 "$BOOT_PID" 2>/dev/null; then
|
|
echo "[-] make boot exited early during boot analysis."
|
|
tail -n 80 "$BOOT_LOG" 2>/dev/null || true
|
|
die "Boot analysis failed: process exited early."
|
|
fi
|
|
|
|
boot_state="$(monitor_boot_log_until "$BOOT_ANALYSIS_TIMEOUT")"
|
|
case "$boot_state" in
|
|
panic)
|
|
echo "[-] Boot analysis: panic detected, stopping VM."
|
|
stop_process_tree "$BOOT_PID"
|
|
BOOT_PID=""
|
|
tail -n 80 "$BOOT_LOG" 2>/dev/null || true
|
|
die "Boot analysis failed: panic detected."
|
|
;;
|
|
bash)
|
|
echo "[+] Boot analysis: bash prompt detected, boot success."
|
|
stop_process_tree "$BOOT_PID"
|
|
BOOT_PID=""
|
|
;;
|
|
exited)
|
|
echo "[-] Boot analysis: VM process exited before success marker."
|
|
BOOT_PID=""
|
|
tail -n 80 "$BOOT_LOG" 2>/dev/null || true
|
|
die "Boot analysis failed: process exited."
|
|
;;
|
|
timeout)
|
|
echo "[-] Boot analysis timeout (${BOOT_ANALYSIS_TIMEOUT}s); stopping VM."
|
|
stop_process_tree "$BOOT_PID"
|
|
BOOT_PID=""
|
|
tail -n 80 "$BOOT_LOG" 2>/dev/null || true
|
|
die "Boot analysis timeout."
|
|
;;
|
|
esac
|
|
}
|
|
|
|
trap cleanup EXIT INT TERM
|
|
|
|
check_platform() {
|
|
[[ "$(uname -s)" == "Darwin" ]] || die "This script supports macOS only"
|
|
|
|
local major
|
|
major="$(sw_vers -productVersion | cut -d. -f1)"
|
|
if [[ -z "$major" || "$major" -lt 15 ]]; then
|
|
die "macOS 15+ required (detected: $(sw_vers -productVersion))"
|
|
fi
|
|
|
|
xcrun -sdk iphoneos --show-sdk-path >/dev/null 2>&1 \
|
|
|| die "iOS SDK not found. Full Xcode is required (Command Line Tools alone does not include the iOS SDK).\n Install Xcode from the App Store, then run:\n sudo xcode-select -s /Applications/Xcode.app/Contents/Developer"
|
|
}
|
|
|
|
install_brew_deps() {
|
|
require_cmd brew
|
|
|
|
local deps=(
|
|
wget gnu-tar openssl@3 ldid-procursus sshpass keystone git-lfs
|
|
python@3.13 libusb ipsw
|
|
)
|
|
|
|
echo "=== Installing Homebrew dependencies ==="
|
|
for pkg in "${deps[@]}"; do
|
|
if brew list --formula "$pkg" >/dev/null 2>&1; then
|
|
echo " $pkg: already installed"
|
|
else
|
|
echo " $pkg: installing"
|
|
brew install "$pkg"
|
|
fi
|
|
done
|
|
echo ""
|
|
}
|
|
|
|
ensure_python_linked() {
|
|
if ! command -v python3.13 >/dev/null 2>&1; then
|
|
local pybin
|
|
pybin="$(brew --prefix python@3.13)/bin"
|
|
export PATH="$pybin:$PATH"
|
|
fi
|
|
|
|
require_cmd python3.13
|
|
}
|
|
|
|
run_make() {
|
|
local label="$1"
|
|
shift
|
|
|
|
echo ""
|
|
echo "=== ${label} ==="
|
|
if [[ -n "${SUDO_PASSWORD:-}" ]]; then
|
|
sudo -A -v >/dev/null 2>&1 || true
|
|
fi
|
|
make "$@"
|
|
}
|
|
|
|
start_boot_dfu() {
|
|
mkdir -p "$LOG_DIR"
|
|
|
|
if [[ -n "$DFU_PID" ]] && kill -0 "$DFU_PID" 2>/dev/null; then
|
|
return
|
|
fi
|
|
|
|
kill_stale_vphone_procs
|
|
check_vm_storage_locks
|
|
|
|
# Remove stale prediction file so load_device_identity waits for the fresh
|
|
# one written by this boot, avoiding an ECID mismatch race.
|
|
rm -f "${VM_DIR_ABS}/udid-prediction.txt"
|
|
|
|
: > "$DFU_LOG"
|
|
echo "[*] Starting DFU boot in background..."
|
|
(make boot_dfu >"$DFU_LOG" 2>&1) &
|
|
DFU_PID=$!
|
|
|
|
sleep 2
|
|
if ! kill -0 "$DFU_PID" 2>/dev/null; then
|
|
echo "[-] make boot_dfu exited early. Last log lines:"
|
|
tail -n 40 "$DFU_LOG" || true
|
|
exit 1
|
|
fi
|
|
|
|
echo "[+] boot_dfu running (pid=$DFU_PID, log=$DFU_LOG)"
|
|
}
|
|
|
|
stop_boot_dfu() {
|
|
if [[ -n "$DFU_PID" ]] && kill -0 "$DFU_PID" 2>/dev/null; then
|
|
echo "[*] Stopping background DFU boot (pid=$DFU_PID)..."
|
|
stop_process_tree "$DFU_PID"
|
|
fi
|
|
DFU_PID=""
|
|
force_release_vm_locks
|
|
}
|
|
|
|
wait_for_post_restore_reboot() {
|
|
local remaining="${POST_RESTORE_KILL_DELAY}"
|
|
local panic_seen=0
|
|
|
|
echo "[*] Restore complete; waiting up to ${POST_RESTORE_KILL_DELAY}s for reboot/panic before stopping DFU..."
|
|
while (( remaining > 0 )); do
|
|
if [[ -f "$DFU_LOG" ]] && grep -Eiq 'panic|kernel panic' "$DFU_LOG"; then
|
|
panic_seen=1
|
|
break
|
|
fi
|
|
if [[ -n "$DFU_PID" ]] && ! kill -0 "$DFU_PID" 2>/dev/null; then
|
|
echo "[*] DFU process exited during post-restore reboot window."
|
|
return
|
|
fi
|
|
sleep 1
|
|
remaining=$(( remaining - 1 ))
|
|
done
|
|
|
|
if (( panic_seen == 1 )); then
|
|
echo "[+] Panic marker observed; stopping DFU now."
|
|
else
|
|
echo "[*] No panic marker observed in ${POST_RESTORE_KILL_DELAY}s; stopping DFU anyway."
|
|
fi
|
|
}
|
|
|
|
wait_for_recovery() {
|
|
local pmd3_python
|
|
pmd3_python="$(find_python_for_pmd3 || true)"
|
|
[[ -x "$pmd3_python" ]] || die "pymobiledevice3 python runtime not found (run: make setup_tools)"
|
|
[[ -f "$PMD3_BRIDGE" ]] || die "Missing bridge script: $PMD3_BRIDGE"
|
|
|
|
echo "[*] Waiting for recovery/DFU endpoint..."
|
|
local i
|
|
for i in {1..90}; do
|
|
if "$pmd3_python" "$PMD3_BRIDGE" recovery-probe --ecid "0x${DEVICE_ECID}" --timeout 2 >/dev/null 2>&1; then
|
|
echo "[+] Device endpoint is reachable"
|
|
return
|
|
fi
|
|
sleep 2
|
|
done
|
|
|
|
echo "[-] Timed out waiting for device endpoint. Last DFU log lines:"
|
|
tail -n 60 "$DFU_LOG" || true
|
|
exit 1
|
|
}
|
|
|
|
start_iproxy() {
|
|
[[ -n "$DEVICE_UDID" ]] || die "Device UDID is empty; cannot resolve iproxy target"
|
|
|
|
choose_ramdisk_ssh_port
|
|
wait_for_iproxy_target_udid
|
|
|
|
if port_is_listening "$RAMDISK_SSH_PORT"; then
|
|
if [[ "$RAMDISK_SSH_PORT_EXPLICIT" == "1" ]]; then
|
|
die "RAMDISK_SSH_PORT ${RAMDISK_SSH_PORT} is already in use"
|
|
fi
|
|
RAMDISK_SSH_PORT="$(pick_random_ssh_port)" \
|
|
|| die "Failed to allocate a free random local SSH forward port"
|
|
fi
|
|
|
|
IPROXY_LOG="${LOG_DIR}/iproxy_${RAMDISK_SSH_PORT}.log"
|
|
mkdir -p "$LOG_DIR"
|
|
: > "$IPROXY_LOG"
|
|
|
|
local pmd3_python
|
|
pmd3_python="$(find_python_for_pmd3 || true)"
|
|
[[ -x "$pmd3_python" ]] || die "pymobiledevice3 python runtime not found (run: make setup_tools)"
|
|
echo "[*] Starting pymobiledevice3 usbmux forward ${RAMDISK_SSH_PORT} -> 22 (target_udid=${IPROXY_TARGET_UDID}, restore_udid=${DEVICE_UDID}, ecid=0x${DEVICE_ECID})..."
|
|
("$pmd3_python" -m pymobiledevice3 usbmux forward --serial "$IPROXY_TARGET_UDID" "$RAMDISK_SSH_PORT" 22 >"$IPROXY_LOG" 2>&1) &
|
|
IPROXY_PID=$!
|
|
|
|
sleep 1
|
|
if ! kill -0 "$IPROXY_PID" 2>/dev/null; then
|
|
echo "[-] iproxy exited early. Log:"
|
|
tail -n 40 "$IPROXY_LOG" || true
|
|
exit 1
|
|
fi
|
|
|
|
echo "[+] iproxy running (pid=$IPROXY_PID, log=$IPROXY_LOG)"
|
|
}
|
|
|
|
wait_for_ramdisk_ssh() {
|
|
local sshpass_bin
|
|
local waited=0
|
|
|
|
[[ "$RAMDISK_SSH_TIMEOUT" == <-> ]] || die "RAMDISK_SSH_TIMEOUT must be an integer (seconds)"
|
|
[[ "$RAMDISK_SSH_INTERVAL" == <-> ]] || die "RAMDISK_SSH_INTERVAL must be an integer (seconds)"
|
|
(( RAMDISK_SSH_TIMEOUT > 0 )) || die "RAMDISK_SSH_TIMEOUT must be > 0"
|
|
(( RAMDISK_SSH_INTERVAL > 0 )) || die "RAMDISK_SSH_INTERVAL must be > 0"
|
|
|
|
sshpass_bin="$(command -v sshpass || true)"
|
|
[[ -x "$sshpass_bin" ]] || die "sshpass not found (run: make setup_tools)"
|
|
|
|
echo "[*] Waiting for ramdisk SSH on ${RAMDISK_SSH_USER}@127.0.0.1:${RAMDISK_SSH_PORT} (timeout=${RAMDISK_SSH_TIMEOUT}s)..."
|
|
while (( waited < RAMDISK_SSH_TIMEOUT )); do
|
|
if [[ -n "$IPROXY_PID" ]] && ! kill -0 "$IPROXY_PID" 2>/dev/null; then
|
|
echo "[-] iproxy process exited while waiting for ramdisk SSH."
|
|
if [[ -n "$IPROXY_LOG" ]]; then
|
|
echo "[-] iproxy log tail:"
|
|
tail -n 40 "$IPROXY_LOG" 2>/dev/null || true
|
|
fi
|
|
die "iproxy exited before ramdisk SSH became ready."
|
|
fi
|
|
|
|
if [[ -f "$DFU_LOG" ]] && grep -Eiq 'panic|kernel panic|stackshot succeeded|panic\.apple\.com' "$DFU_LOG"; then
|
|
echo "[-] Detected panic markers in boot_dfu log while waiting for ramdisk SSH."
|
|
echo "[-] boot_dfu log tail:"
|
|
tail -n 80 "$DFU_LOG" 2>/dev/null || true
|
|
die "Ramdisk boot appears to have panicked before SSH became ready."
|
|
fi
|
|
|
|
if [[ -n "$DFU_PID" ]] && ! kill -0 "$DFU_PID" 2>/dev/null; then
|
|
echo "[-] boot_dfu process exited while waiting for ramdisk SSH."
|
|
echo "[-] boot_dfu log tail:"
|
|
tail -n 80 "$DFU_LOG" 2>/dev/null || true
|
|
die "DFU boot exited before ramdisk SSH became ready."
|
|
fi
|
|
|
|
if "$sshpass_bin" -p "$RAMDISK_SSH_PASS" ssh \
|
|
-o StrictHostKeyChecking=no \
|
|
-o UserKnownHostsFile=/dev/null \
|
|
-o PreferredAuthentications=password \
|
|
-o ConnectTimeout=5 \
|
|
-q \
|
|
-p "$RAMDISK_SSH_PORT" \
|
|
"${RAMDISK_SSH_USER}@127.0.0.1" "echo ready" >/dev/null 2>&1
|
|
then
|
|
echo "[+] Ramdisk SSH is ready"
|
|
return
|
|
fi
|
|
|
|
if (( waited == 0 || waited % 10 == 0 )); then
|
|
echo " waiting... ${waited}s elapsed"
|
|
fi
|
|
|
|
sleep "$RAMDISK_SSH_INTERVAL"
|
|
(( waited += RAMDISK_SSH_INTERVAL ))
|
|
done
|
|
|
|
echo "[-] Timed out waiting for ramdisk SSH readiness."
|
|
echo "[-] Identity context: restore_udid=${DEVICE_UDID}, ecid=0x${DEVICE_ECID}, iproxy_target_udid=${IPROXY_TARGET_UDID}"
|
|
echo "[-] USBMux IDs at timeout:"
|
|
print_usbmux_udids
|
|
if [[ -n "$IPROXY_LOG" ]]; then
|
|
echo "[-] iproxy log tail:"
|
|
tail -n 40 "$IPROXY_LOG" 2>/dev/null || true
|
|
fi
|
|
echo "[-] boot_dfu log tail:"
|
|
tail -n 60 "$DFU_LOG" 2>/dev/null || true
|
|
die "Ramdisk SSH did not become ready in ${RAMDISK_SSH_TIMEOUT}s."
|
|
}
|
|
|
|
stop_iproxy() {
|
|
if [[ -n "$IPROXY_PID" ]] && kill -0 "$IPROXY_PID" 2>/dev/null; then
|
|
echo "[*] Stopping iproxy (pid=$IPROXY_PID)..."
|
|
stop_process_tree "$IPROXY_PID"
|
|
fi
|
|
IPROXY_PID=""
|
|
}
|
|
|
|
parse_args() {
|
|
local arg
|
|
for arg in "$@"; do
|
|
case "$arg" in
|
|
--jb)
|
|
JB_MODE=1
|
|
;;
|
|
--dev)
|
|
DEV_MODE=1
|
|
;;
|
|
--skip-project-setup)
|
|
SKIP_PROJECT_SETUP=1
|
|
;;
|
|
-h|--help)
|
|
cat <<'EOF'
|
|
Usage: setup_machine.sh [--jb] [--dev] [--skip-project-setup]
|
|
|
|
Options:
|
|
--jb Use jailbreak firmware patching + jailbreak CFW install.
|
|
--dev Use dev firmware patching + dev CFW install.
|
|
--skip-project-setup Skip setup_tools/build stage.
|
|
|
|
Environment:
|
|
NONE_INTERACTIVE=1 Auto-continue first-boot prompts + run final boot analysis.
|
|
SUDO_PASSWORD=... Preload sudo credential via askpass.
|
|
EOF
|
|
exit 0
|
|
;;
|
|
*)
|
|
die "Unknown argument: $arg"
|
|
;;
|
|
esac
|
|
done
|
|
}
|
|
|
|
main() {
|
|
parse_args "$@"
|
|
if parse_bool "$NONE_INTERACTIVE_RAW"; then
|
|
NONE_INTERACTIVE=1
|
|
fi
|
|
setup_sudo_noninteractive
|
|
|
|
local fw_patch_target="fw_patch"
|
|
local cfw_install_target="cfw_install"
|
|
local mode_label="base"
|
|
|
|
if [[ "$JB_MODE" -eq 1 && "$DEV_MODE" -eq 1 ]]; then
|
|
die "--jb and --dev are mutually exclusive"
|
|
fi
|
|
|
|
if [[ "$JB_MODE" -eq 1 ]]; then
|
|
fw_patch_target="fw_patch_jb"
|
|
cfw_install_target="cfw_install_jb"
|
|
mode_label="jailbreak"
|
|
elif [[ "$DEV_MODE" -eq 1 ]]; then
|
|
fw_patch_target="fw_patch_dev"
|
|
cfw_install_target="cfw_install_dev"
|
|
mode_label="dev"
|
|
fi
|
|
|
|
echo "[*] setup_machine mode: ${mode_label}, project_setup=$([[ "$SKIP_PROJECT_SETUP" -eq 1 ]] && echo "skip" || echo "run"), non_interactive=${NONE_INTERACTIVE}"
|
|
|
|
if [[ "$SKIP_PROJECT_SETUP" -eq 1 ]]; then
|
|
echo ""
|
|
echo "=== Project setup ==="
|
|
echo "[*] Skipping setup_tools/build"
|
|
else
|
|
check_platform
|
|
install_brew_deps
|
|
ensure_python_linked
|
|
|
|
run_make "Project setup" setup_tools
|
|
run_make "Project setup" build
|
|
fi
|
|
|
|
# Activate venv so all child scripts (cfw_install, patchers, etc.) use the
|
|
# project Python with capstone/keystone/pyimg4 installed, not the bare system python3.
|
|
export PATH="$PROJECT_ROOT/.venv/bin:$PATH"
|
|
|
|
run_make "Firmware prep" vm_new
|
|
run_make "Firmware prep" fw_prepare
|
|
run_make "Firmware patch" "$fw_patch_target"
|
|
|
|
echo ""
|
|
echo "=== Restore phase ==="
|
|
start_boot_dfu
|
|
load_device_identity
|
|
wait_for_recovery
|
|
run_make "Restore" restore_get_shsh RESTORE_UDID="$DEVICE_UDID" RESTORE_ECID="0x$DEVICE_ECID"
|
|
run_make "Restore" restore RESTORE_UDID="$DEVICE_UDID" RESTORE_ECID="0x$DEVICE_ECID"
|
|
wait_for_post_restore_reboot
|
|
stop_boot_dfu
|
|
echo "[*] Waiting ${POST_KILL_SETTLE_DELAY}s for cleanup before ramdisk stage..."
|
|
sleep "$POST_KILL_SETTLE_DELAY"
|
|
|
|
echo ""
|
|
echo "=== Ramdisk + CFW phase ==="
|
|
start_boot_dfu
|
|
load_device_identity
|
|
wait_for_recovery
|
|
run_make "Ramdisk" ramdisk_build RAMDISK_UDID="$DEVICE_UDID"
|
|
echo "[*] Ramdisk identity context: restore_udid=${DEVICE_UDID} ecid=0x${DEVICE_ECID}"
|
|
run_make "Ramdisk" ramdisk_send IRECOVERY_ECID="0x$DEVICE_ECID" RAMDISK_UDID="$DEVICE_UDID"
|
|
start_iproxy
|
|
|
|
wait_for_ramdisk_ssh
|
|
|
|
run_make "CFW install" "$cfw_install_target" SSH_PORT="$RAMDISK_SSH_PORT"
|
|
stop_boot_dfu
|
|
stop_iproxy
|
|
|
|
echo ""
|
|
echo "=== First boot ==="
|
|
if [[ "$NONE_INTERACTIVE" -eq 0 ]]; then
|
|
read -r "?[*] press Enter to start VM, after the VM has finished booting, press Enter again to finish last stage"
|
|
else
|
|
echo "[*] NONE_INTERACTIVE=1: auto-starting first boot"
|
|
fi
|
|
|
|
start_first_boot
|
|
|
|
if [[ "$NONE_INTERACTIVE" -eq 0 ]]; then
|
|
read -r "?[*] Press Enter once the VM is fully booted"
|
|
else
|
|
wait_for_first_boot_prompt_auto
|
|
fi
|
|
send_first_boot_commands
|
|
|
|
echo "[*] Commands sent. Waiting for VM shutdown..."
|
|
wait "$BOOT_PID"
|
|
BOOT_PID=""
|
|
|
|
exec {BOOT_FIFO_FD}>&- || true
|
|
BOOT_FIFO_FD=""
|
|
rm -f "$BOOT_FIFO" || true
|
|
BOOT_FIFO=""
|
|
|
|
if [[ "$JB_MODE" -eq 1 ]]; then
|
|
echo ""
|
|
echo "=== JB Finalize ==="
|
|
echo "[*] JB finalization will run automatically on first normal boot"
|
|
echo " via /cores/vphone_jb_setup.sh (LaunchDaemon)."
|
|
echo " Monitor progress via vphoned file browser: /var/log/vphone_jb_setup.log"
|
|
fi
|
|
|
|
echo ""
|
|
echo "=== Done ==="
|
|
echo "Setup completed."
|
|
|
|
echo "=== Boot analysis ==="
|
|
run_boot_analysis
|
|
}
|
|
|
|
main "$@"
|