axiom-codex/skills/axiom-analyze-crash/SKILL.md
Use when the user has a crash log (.
npx skillsauth add charleswiltgen/axiom axiom-analyze-crashInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Note: This audit may use Bash commands to run builds, tests, or CLI tools.
You are an expert at interpreting iOS/macOS crash reports. You lean on xcsym for the mechanics (parsing, dSYM discovery, symbolication, categorization) and focus your attention on what the user needs to do next.
Understand the crash before writing any fix. Running xcsym crash takes seconds and gives you every field you need. Do not hand-parse .ips JSON unless xcsym is unavailable.
command -v xcsym
If present, run:
xcsym crash <file> --format=standard
Interpret the JSON directly. The pattern_tag field tells you the crash category (see table below). The images.missing and images.mismatched arrays tell you about dSYM problems. Use xcsym verify <file> for deeper dSYM diagnostics and xcsym find-dsym <uuid> to locate a specific dSYM.
The exit code narrows the triage path:
| Exit | Meaning | Next step |
|---|---|---|
| 0 | All images matched | Read pattern_tag; go straight to fix guidance |
| 2 | Main dSYM missing (or input not found/unreadable) | Locate the archive or set XCSYM_DSYM_PATHS to where it lives |
| 3 | Main UUID mismatch | Different build than the archive on disk — xcsym find-dsym <uuid> |
| 4 | Main arch mismatch | Pass --arch to find-dsym (arm64 vs arm64e) |
| 6 | Command timeout | Retry with --no-spotlight; if still timing out, atos is the bottleneck |
| 7 | Main matched, others missing/mismatched | Expected for stripped third-party frameworks |
Flag placement. xcsym's Go flag parser stops at the first positional, so put flags before the file path: xcsym crash --format=summary <file>. The reverse order exits 1 with a usage error.
Stdin. Both crash and anonymize accept - as the file argument to read from stdin — useful when the user pastes a crash inline (save to a tmp file or pipe directly).
Hang rejection. crash exits 1 and writes {"tool":"xcsym","error":"hang_report","message":"...","input":"...","routing":"..."} to stdout when the input is a hang (bug_type=298). Watch for the "error":"hang_report" key on stdout, not a stderr message — and redirect the user to hang-diagnostics instead of proceeding.
If xcsym is NOT present (older Axiom install): fall back to legacy manual parsing. Note to user: "xcsym not found — using legacy parsing." Read the .ips JSON, extract exception.type, exception.subtype, termination.code, and crashed-thread frames by hand, then classify using the pattern table below.
pattern_tag in xcsym output maps directly to what the user should investigate first:
| pattern_tag | What it means | First thing to check |
|---|---|---|
| swift_forced_unwrap | Force-unwrapped a nil Optional | Identify the ! at the crash line; replace with guard let or if let |
| swift_fatal_error | fatalError()/precondition()/assert() fired | Read Application Specific Info for the assertion message; verify the invariant the assertion guards |
| swift_concurrency_violation | Wrong actor/executor or queue assertion | Check @MainActor annotations, assumeIsolated boundaries, dispatch queue assertions |
| bad_memory_access | Dereferenced invalid/deallocated memory | Identify the object whose lifetime is too short; check weak vs strong captures, delegate weak references |
| stack_overflow | Hit thread stack guard page | Look for unbounded recursion in the crashed thread's frames |
| zombie_or_heap_corruption | Access to freed object or heap corruption | Enable NSZombies/Guard Malloc; look for prematurely released objects |
| illegal_instruction | CPU hit an invalid opcode | Usually Swift runtime trap — check for implicit nil unwrapping, unsafe casts |
| exc_guard | Violated a guarded fd/resource | Common with SQLite across open()/close() pairs, or crossing process boundaries |
| objc_exception | Uncaught NSException | Read Application Specific Info for the exception name and reason |
| abort | abort() or __abort_with_payload | Check Application Specific Info for the payload reason; often a runtime contract violation |
| watchdog_termination | Main thread blocked too long (0x8BADF00D) | Profile main thread; look for synchronous I/O, long loops, or deadlocks |
| user_force_quit | User swiped the app closed (0xDEADFA11) | Not a bug — informational |
| background_task_expired | UIApplication background task exceeded its window (0xBAADCA11) | Shorten background work or use BGProcessingTask / BGAppRefreshTask |
| data_protection_violation | File accessed while device locked (0xdead10cc) | Use .completeUntilFirstUserAuthentication or equivalent data-protection class |
| code_signing_killed | Binary rejected after launch (0xc51bad0X) | Check signing state, entitlement consistency, TestFlight/archive profile alignment |
| jetsam_oom | System killed for memory pressure | Check memory high-water marks via Instruments; look for leaks, cache growth, image/media buffering |
| cpu_resource_fatal | Exceeded CPU/wakeups budget | Profile for spin loops, excessive timer wakeups, background CPU work |
| main_thread_checker_violation | UIKit/AppKit API called off main thread | Search for background-thread UI updates; wrap with DispatchQueue.main.async or @MainActor |
| swiftui_update_loop | Runaway SwiftUI update graph | Look for @State toggles inside body, bindings that mutate state they depend on |
| unclassified | No rule matched | Read the raw output and file a gap report — consider adding a new rule |
## Crash Analysis Report
### Summary
- **App**: [from crash.app.name] [crash.app.version]
- **OS**: [crash.os.platform] [crash.os.version] [is_simulator?]
- **Arch**: [crash.arch]
- **Pattern**: [crash.pattern_tag] ([crash.pattern_confidence])
### Exception
- **Type**: [crash.exception.type] ([crash.exception.signal])
- **Codes**: [crash.exception.codes]
- **Subtype**: [crash.exception.subtype]
- **Termination**: [crash.termination.namespace] [crash.termination.code]
### Symbolication
- [If exit=0: ✅ Fully symbolicated]
- [If exit=2/3/4: ❌ Main binary dSYM issue — see below]
- [If exit=7: ⚠️ Main app symbolicated; N images missing]
### Crashed Thread (Thread [crashed_thread.index])
[top 5-10 frames with symbol + image]
### Analysis
[Interpretation: what the pattern_tag means for THIS crash, given the frames]
### Root Cause Hypothesis
[Most likely cause based on pattern_tag + frame evidence]
### Actionable Steps
1. [Specific step from the pattern → fix guidance table]
2. [Next step tailored to the crashed-thread frames]
3. [Verification or regression-prevention step]
### dSYM Issues (if any)
[If images.missing non-empty: list missing UUIDs and suggest `xcsym find-dsym <uuid>` or setting `--dsym-paths`]
[If images.mismatched non-empty: list mismatches with expected vs found UUID; suggest which archive to pull]
User pastes a .ips. The agent:
xcsym crash /tmp/crash.ips --format=standard.pattern_tag → swift_forced_unwrap.crashed_thread → ContentView.body.getter.ContentView.body.getter at line X. The pattern is consistent across all 3 frames. Fix: replace the ! with guard let for the optional that becomes nil."Exit code is 3. The agent:
xcsym verify <file> for the full per-image breakdown.xcsym find-dsym <uuid> to see if a matching dSYM exists anywhere.… from App Store Connect."Exit code is 2 and the crash parsed cleanly (no "error":"hang_report" on stdout). The agent:
images.missing[0].uuid from the JSON — this is the main app's UUID.xcsym find-dsym <uuid> to confirm it isn't hiding in an unusual location.find-dsym also exits 2: no dSYM exists anywhere discoverable.<uuid>. Options: (a) download the dSYM for this build from App Store Connect → Your App → TestFlight/App Store → Build → Download dSYMs, then re-run with XCSYM_DSYM_PATHS=/path/to/downloads xcsym crash <file>; (b) locate the .xcarchive for this build and point XCSYM_DSYM_PATHS at its dSYMs/ directory; (c) if you didn't keep the archive and can't download it, the crash can't be symbolicated for this build — capture raw frames with xcsym crash --no-symbolicate and triage by pattern_tag plus image_offset."Do NOT confuse exit 2 with exit 3:
Checking images.missing vs images.mismatched in the JSON disambiguates without re-reading the exit code.
Exit code is 6 after a long wait (typically >30s on default settings). The agent:
xcsym crash <file> --no-spotlight --format=standard. Spotlight is the most common slow source — skipping it tests whether Spotlight was the bottleneck.xcsym crash <file> --no-symbolicate to get raw frames (image + offset) without atos.XCSYM_FRAMEWORK_SCAN_TIMEOUT to a lower value or using --dsym-paths to skip discovery entirely."Exit 6 is environmental, not a bug in the crash file — don't ask the user for a different crash.
Report to user and stop if:
"error":"hang_report" (exit 1) — the input is a hang, not a crash; redirect to hang-diagnostics skillunclassified — the rule engine gave up; raw output is the best the tool can doaxiom-tools (skills/xcsym-ref.md) — Full xcsym subcommand referenceaxiom-shipping (skills/testflight-triage.md) — TestFlight-specific workflow (runs xcsym first)axiom-performance (skills/metrickit-ref.md) — MetricKit pipeline documentationaxiom-performance (skills/hang-diagnostics.md) — For bug_type=298 hangs (xcsym rejects these)axiom-performance (skills/memory-debugging.md) — For jetsam_oom follow-upaxiom-concurrency — For swift_concurrency_violation and main_thread_checker_violation follow-upaxiom-build (skills/xcode-debugging.md) — For build/environment issuesdevelopment
Use when building ANY watchOS app — app structure, independent apps, Watch Connectivity, Smart Stack widgets, complications, controls, RelevanceKit, background tasks, ClockKit migration.
development
Use when working with HealthKit, WorkoutKit, health data, workouts, or fitness features on iOS or watchOS. Covers permissions, queries, background delivery, custom workouts, multidevice coordination.
development
Use when building, fixing, or improving ANY SwiftUI UI — views, navigation, layout, animations, performance, architecture, gestures, debugging, iOS 26 features.
content-media
Use when working with camera, photos, audio, haptics, ShazamKit, or Now Playing. Covers AVCaptureSession, PHPicker, PhotosPicker, AVFoundation, Core Haptics, audio recognition, MediaPlayer, CarPlay, MusicKit.