Files
sx/tests/debug_stepping_smoke.sh
agra b2ebf774bc ERR/E3.0 (slice 3e rung 2): iOS-simulator stepping verified
Closes out E3's stepping-verification ladder to the extent possible
headlessly.

- Verified `sx build --target ios-sim --emit-obj` produces an
  arm64-ios-simulator Mach-O that runs under `simctl spawn` and steps
  in lldb (the backtrace shows a dyld_sim frame — the sim runtime).
- Verified the device-applicable .dSYM path: dsymutil collects the
  DWARF, and after removing the .o lldb still resolves source via the
  .dSYM.
- debug_stepping_smoke.sh gains an optional iOS-sim rung that reuses an
  already-booted simulator (never boots one — single-sim policy) and
  exercises the .dSYM path; skips cleanly when no sim is booted.
- docs/debugger.md: rungs 1-2 marked verified; the iOS-device rung is
  documented as a manual checklist (needs hardware + get-task-allow
  signing) — no compiler gap, --emit-obj + standard Apple tools suffice.

E3 is functionally complete and verified across macOS + iOS-simulator.
2026-06-01 16:10:33 +03:00

76 lines
3.2 KiB
Bash
Executable File

#!/usr/bin/env bash
# Debug-stepping smoke (ERR E3.0 slice 3e, rung 1: macOS native).
#
# Verifies the DWARF emitted by `sx build --emit-obj` actually drives
# source-level stepping in lldb — the deep-debug half of the trace story.
# NOT part of `run_examples.sh` (it needs `lldb`, and is macOS-specific via the
# debug-map → kept `.o`). Run manually: bash tests/debug_stepping_smoke.sh
#
# Exit 0 = lldb resolved a file:line breakpoint + a source-mapped backtrace.
set -u
ROOT_DIR="$(cd "$(dirname "$0")/.." && pwd)"
SX="$ROOT_DIR/zig-out/bin/sx"
TMP="$ROOT_DIR/.sx-tmp"
SRC="$TMP/dbg_smoke.sx"
BIN="$TMP/dbg_smoke"
if ! command -v lldb >/dev/null 2>&1; then
echo "SKIP: lldb not found (macOS/Xcode tools required)"
exit 0
fi
mkdir -p "$TMP"
cat > "$SRC" <<'EOF'
add :: (a: s32, b: s32) -> s32 {
c := a + b;
return c;
}
main :: () -> s32 {
return add(40, 2);
}
EOF
"$SX" build --emit-obj "$SRC" -o "$BIN" >/dev/null 2>&1 || { echo "FAIL: build"; exit 1; }
# Breakpoint on the `return c;` line; expect lldb to resolve it + a backtrace
# mapping both frames to dbg_smoke.sx.
out=$(cd "$ROOT_DIR" && lldb --batch \
-o "breakpoint set --file dbg_smoke.sx --line 3" \
-o "run" -o "bt" -o "quit" "$BIN" 2>&1)
fail=0
echo "$out" | grep -q "dbg_smoke.sx:3" || { echo "FAIL[macos]: breakpoint did not resolve to dbg_smoke.sx:3"; fail=1; }
echo "$out" | grep -q "add at dbg_smoke.sx:3" || { echo "FAIL[macos]: stopped frame not source-mapped"; fail=1; }
echo "$out" | grep -q "main at dbg_smoke.sx:6" || { echo "FAIL[macos]: caller frame not source-mapped"; fail=1; }
[[ $fail -eq 0 ]] && echo "ok[macos]: lldb stepped sx source (debug map -> kept .o)"
# Rung 2 (iOS simulator) — runs ONLY against an ALREADY-booted simulator; it
# never boots one itself (use a single sim — boot one yourself if you want this
# rung). Builds for ios-sim, collects a .dSYM (the device-applicable artifact),
# and steps in the sim. Skipped if no booted sim / no ios-sim SDK.
sim_booted=$(xcrun simctl list devices booted 2>/dev/null | grep -c Booted || true)
if [[ "${sim_booted:-0}" -gt 0 ]] && xcrun --sdk iphonesimulator --show-sdk-path >/dev/null 2>&1; then
"$SX" build --target ios-sim --emit-obj "$SRC" -o "$BIN" >/dev/null 2>&1 || { echo "FAIL[ios-sim]: build"; fail=1; }
dsymutil "$BIN" -o "$BIN.dSYM" >/dev/null 2>&1 # the .app ships a .dSYM, not the .o
rm -f "$TMP/main.o" # force resolution via the .dSYM
sout=$(cd "$ROOT_DIR" && timeout 120 lldb --batch \
-o "breakpoint set --file dbg_smoke.sx --line 3" \
-o "run" -o "bt" -o "quit" "$BIN" 2>&1)
if echo "$sout" | grep -q "add at dbg_smoke.sx:3" && echo "$sout" | grep -q "main at dbg_smoke.sx:6"; then
echo "ok[ios-sim]: lldb stepped sx source in the simulator (via .dSYM)"
else
echo "FAIL[ios-sim]: lldb did not resolve in the simulator"; fail=1
echo "--- ios-sim lldb output ---"; echo "$sout"
fi
rm -rf "$BIN" "$BIN.dSYM"
else
echo "skip[ios-sim]: no booted simulator (boot one to exercise this rung)"
fi
# Rung 3 (iOS device) is a documented manual checklist — see docs/debugger.md.
rm -f "$SRC" "$BIN" "$TMP/main.o"
[[ $fail -eq 0 ]] && exit 0
exit 1