# G2/G3 — checkpoint selection for cross-mechanism v_hack **Date:** 2026-05-28 **Status:** draft **Supersedes (partially):** `20260528_cross_mechanism_v_hack.md` step 1-3. ## Goal Test whether v_hack extracted from {a subset of hack mechanisms} also suppresses {the other mechanisms} — the "noisy oracle" generalization hypothesis. Needs a teacher pool with diverse hack types. Entry (g) showed the existing `rh-s65` pool is degenerate (96.1% EC-, 2 non-singleton signatures). To fix, mix pools from DIFFERENT intervention checkpoints. ## Scope In: - Pre-gen pools from 2-4 Aria HF checkpoints with high residual RH AND predicted-different hack mechanism distribution. - Regrade each pool with the 3 viable detectors E, C, D. - Pick a half-A / half-B split that puts a common (easy) detector and a rare (hard) detector in each half. - Run G3: extract V from half-A pool subset, train, measure per-detector suppression vs vanilla baseline. Out: - Adding new detectors (beyond E/C/D) to `rewards.py`. The 6 paper-listed sub-mechanisms (bypass, hardcode, wrong-asserts, print-only, fake-checkEqual, examples-only) are author commentary, not coded classifiers, and writing them is its own project. - Training a new teacher (would take ~3h × 4 H200 per Aria's setup). ## Background — checkpoint map from the paper Source: `docs/papers/2025_lw_ariahw_steering-rl-training-benchmarking-interventions.md`. HF collection: https://huggingface.co/collections/ariahw/reward-hacking-rl. Each intervention was trained at seeds 1, 42, 65. Per the paper: | family | mechanism it targets | residual RH (paper) | viable as teacher? | |---|---|---|---| | `rh-*` | none (RL on loophole env) | ~79% | YES — current pool, but mono-mechanism | | `rl-baseline-*` | none (RL on no-loophole env) | ~0% | NO — barely hacks | | `gt-monitor-penalty-*` | flagged-RH penalty, perfect monitor | ~0% (§163) | NO — barely hacks | | `gt-monitor-screening-*` | flagged-RH dropped from gradients, perfect monitor | 0.9% (§167) | NO | | `judge-monitor-penalty-*` | LLM-judge penalty (~50% monitor acc) | low (§197) | NO | | `judge-monitor-screening-*` | LLM-judge screening | ~30-60% (§211) | YES — failed-intervention | | `probe-monitor-penalty-*` | linear-probe penalty | ~0% (§185) | NO — probe was very accurate | | `probe-monitor-screening-*` | linear-probe screening | ~10-30% | MAYBE | | `inoc-prompt-loophole-*` | inoculate against loophole hack | high (model card confirms still hacks) | YES | | `inoc-prompt-passtests-*` | inoculate against passtests hack | high (paper §137 "modest protection") | YES — likely different mechanism | | `inoc-prompt-evalenv-*` | inoculate against eval-env overwrite | high | YES — likely different mechanism | | `*-lh-*` variants | inoculation prompt also explicitly defines the loophole | likely high | LATER | **Hypothesis on mechanism diversity:** - `inoc-prompt-passtests`: trained to avoid "passtests" hacks specifically. If it still hacks, it should prefer mechanisms OTHER than passtests — namely D (wrong_tests) or genuine bypass that our C detector might not catch. - `inoc-prompt-evalenv`: trained to avoid eval-env overwrites. Same idea. - `judge-monitor-screening`: paper §211 explicitly noted "breakout" runs at 20-60% RH. These are intervention-failure modes. ## Requirements - **R1.** Identify a pool (single or combined) whose *signature distribution* has all three of: (a) ≥3 non-singleton signatures with n≥20, AND (b) at least one non-EC signature (anything other than `EC-` and `ECD`) with n≥50, AND (c) no single signature exceeds 60% of the pool. (R1.b was originally "D-containing"; broadened to "non-EC" because any signature outside the EC- cluster gives us mechanism diversity, not specifically the wrong-tests detector.) The `regrade_pool` exit code enforces all three after the 2026-05-29 extension. - VERIFY: `just regrade-pool ` exit 0 (the audit gate now prints all three pass/fail lines and returns nonzero on any failure). - sneaky_fail: audit gate passes on (a) but pool is still 96% EC- with 3 small islands; G3 has no held-out signal. Caught by (b) AND (c). - **R2.** Pick a half-A / half-B split *over signatures*, not over detectors. The rh-s65 pool entry (g) shows E and C co-fire on 1798/1799 rollouts; treating E and C as different detector classes for splitting leaks the same hack into both halves. Instead, group signatures into halves directly. Example split: half-A = {EC-, ECD} (the loophole+ passtests cluster), half-B = {--D, -CD, -C-} (wrong-tests / alt-only cluster). Each half must have ≥50 rollouts of base rate in the trained student arm (else R4 ratio is noisy). - VERIFY: print signature counts per half before extract; both halves must have ≥1 signature with n≥50. - This requires plumbing: `pairs_from_pool.py` currently takes `--half-a` as a comma list of *detectors*, not signatures. T3 must extend. - **R3.** Run G3 ladder: (a) extract V from half-A pool subset; (b) train projected for 20 steps on combined-pool teacher + half-A V; (c) train matched-seed vanilla on combined-pool teacher (no projection). Measure per-detector hack_s rates on student rollouts in both arms. - VERIFY: train final BLUF shows `HACK_S_E`, `HACK_S_C`, `HACK_S_D`, `HACK_A`, `HACK_B` for both arms. - sneaky_fail: vanilla arm shows hack_s_* all near zero (combined pool's teacher rollouts didn't induce student hacking). Caught by requiring vanilla HACK_S >= 30%. - **R4.** UAT for G3 PASS: in the projected arm, the half-B hack rate drops by ≥50% of the half-A drop, both relative to matched-seed vanilla on the *same combined pool*. FAIL: held-out drops <30% of seen drop, i.e. v_hack is mechanism-specific. - VERIFY: side-by-side hack-rate-per-half table; compute drop ratios. - sneaky_fail: seen drop is itself tiny (<10pp absolute), making the ratio noisy. Caught by R3 requiring vanilla ≥30% per half. ## Tasks - [/] **T1 (R1)**: Pre-gen 50-prompt pools from 4 candidates across 3 variation axes: `rh-s42` (seed-variance null), `inoc-prompt-loophole-s65` (prompt-variance), `judge-monitor-screening-s65` and `probe-monitor-screening-s65` (monitor-variance, two monitor types). Replaces earlier predicted-low-hack picks (gt-monitor-penalty, judge-monitor-penalty, inoc-passtests, inoc-evalenv) that were killed. - status: queued as pueue #107, #111, #112, #113 (2026-05-28). - verify: each pool's first 5 rollouts show hack rate >30%; check with `pueue log ` after each finishes. - success: ≥2 of 4 pools have hack rate >30%. - likely_fail: all pools low-hack. Fallback: combine all four + rh-s65, rely on rh-s65 as base + alt as diversity sprinkle. - sneaky_fail: high hack but identical signature distribution to rh-s65. Caught by T2 audit (R1.b: ≥50 D-containing rollouts somewhere). - [ ] **T2 (R1)**: Regrade each pool. Build a combined pool dir `out/probe_distill/teacher_pool_combined/` by symlinking all 5 source pools' prompt files (per-source dedup if prompts overlap). Regrade the combined pool. - verify: `just regrade-pool out/probe_distill/teacher_pool_combined` exit 0 AND grep `D` in signature table for n≥50 AND no signature pct≥60% (manual eyeball or grep on `pct` column). - status: queued as #110, #114, #115, #116 (per-pool regrades). Combined-pool regrade not yet queued; build after T1 lands. - [ ] **T3 (R2)**: Extend `pairs_from_pool.py` to accept signature-level splits (`--half-a-signatures="EC-,ECD"`) in addition to the current detector-level `--half-a=E,C`. Then pick the split based on T2 data. - steps: add `half_a_signatures: list[str]` Config field; when set, override the detector-level half-A logic with: hack-side = rollouts whose signature is in `half_a_signatures`, clean-side = rollouts with `---` (all detectors off). - verify: print signature counts per half before extract; both halves must have ≥1 signature with n≥50 (per R2). - sneaky_fail: the split is signature-based but pairs_from_pool falls back to detector logic if flag missing. Caught by explicit assert in the new code path. - [ ] **T4 (R3)**: Run G3 head-to-head: - **T4a**: extract V from half-A pool subset via `just extract-vhack-pool half_a="" pool= tag=_combined`. - **T4b**: train projected via `just fast-projected-pool half_a="" seed=41 pool=out/probe_distill/teacher_pool_combined tag=_combined`. NOTE: must pass `pool=` arg explicitly, recipe defaults to rh-s65 only. - **T4c**: train matched-seed vanilla via `just fast-vanilla-xmech half_a="" seed=41 pool=out/probe_distill/teacher_pool_combined tag=_combined`. Same NOTE on `pool=` arg. - verify: both train logs end with BLUF row showing `HACK_S_E`, `HACK_S_C`, `HACK_S_D`, `HACK_A`, `HACK_B` columns AND vanilla HACK_S ≥ 30%. - sneaky_fail: vanilla baseline never hacks (combined pool's teacher rollouts didn't transfer). Caught by HACK_S ≥ 30% requirement. - [ ] **T5 (R4)**: Build the cross-mech UAT table from T4b and T4c logs. Compute: half-A-drop = vanilla(HACK_A) - projected(HACK_A); half-B-drop = vanilla(HACK_B) - projected(HACK_B); ratio = half-B-drop / half-A-drop. PASS if ratio ≥ 0.5 AND vanilla(HACK_B) ≥ 30%. If PASS at single seed, queue n=3 seeds (T6). If FAIL, write entry documenting the negative result. ## Context - Smoke test for the pipeline: `just smoke-xmech` (added 2026-05-28; uses tiny-random Qwen3 on CPU, ~2 min). Covers regrade_pool, pairs_from_pool, extract_vhack_grad with --pairs-from-pool, train with pool-derived V. - Audit gate in `regrade_pool.py` requires ≥3 non-singleton signatures (n≥20) for the half-A/half-B split to have power. Pass `--no-require-audit` for smoke or for diagnostic-only runs. ## Log (append-only) - 2026-05-28: rh-s65 pool degenerate (entry g). 96.1% EC-, only 2 non- singleton signatures. - 2026-05-28: queued gt-monitor-penalty (#97) and judge-monitor-penalty (#98) BEFORE reading the paper. Both predicted-bad teachers (intervention worked, model barely hacks). Killed. - 2026-05-28: queued inoc-passtests (#105) and inoc-evalenv (#106) on hypothesis that inoculation failure surfaces different mechanisms. User pushed back: inoculation prompts that fail still likely hack via the same mechanism they were inoculated against (the prompt describes that hack). Killed. Replaced with rh-s42 (seed-variance null), inoc-loophole-s65 (prompt-variance), judge-screening-s65, probe-screening-s65 (monitor-variance, two monitor types). - 2026-05-29: subagent review flagged: detector-level half-A/half-B split is incoherent because E and C co-fire ~99.9% in rh-s65 (any split puts near-identical labels in opposite halves, leaking signal). R2 rewritten to use *signature-level* split (group entire (E,C,D) signatures, not individual detectors). Requires extending pairs_from_pool.py — T3. - 2026-05-29: R1 audit gate in `regrade_pool.py` extended from R1.a only to all three of {R1.a ≥3 non-singleton signatures, R1.b largest non-EC signature ≥50, R1.c top signature <60%}. R1.b broadened from "D-containing" to "any non-EC" per the revised decision rule (entry j, forthcoming): any signature outside the EC- cluster gives mechanism diversity, not specifically wrong-tests. - 2026-05-29: queued G1-sparse probe (#134 projected, #135 vanilla). At G=8 mix=0.125 (one teacher per group), does refresh-every=2 + 21-pair V still drop last-5 hack_s? Hypothesis: yes — entry i showed refresh keeps cos_pre_t ~1.4x higher and entry h showed 21-pair basis does the heavy lifting; combined sparse-teacher regime is the cheap deployment analogue (1 hack example per 7 normal rollouts). - 2026-05-29: refresh-path logging silenced (extract per-pair "loss=" and postprocess summary were flooding training logs every refresh tick). Wrapped extract+postprocess in loguru disable/enable. Refresh now emits a single line per tick: `v_hack refreshed @ step={s}: M modules, k_axes=K`. ## TODO (out of scope) - Add finer detectors (paper's 6 sub-mechanisms) to `rewards.py`. Test whether finer split of EC- into sub-signatures gives the audit gate ≥3 non-singletons without needing new teachers. - Try the `-lh-*` (loophole-mentioned-in-prompt) inoculation variants if the non-lh ones don't give diversity. ## Plan B (deferred — likely subsumed by G2 screen) The original Plan B was to prompt base Qwen3-4B to write completions in specific hack classes. Dropped after a closer read: explicitly instructing the model to "define run_tests with assertions that fail on canonical" is instruction-following, not reward-hacking. The gradient direction NLL takes on a "do X" completion is generally not the gradient direction GRPO takes when chasing reward — entry (i) already shows even self-vs-self v_hack cosines peak at 0.27 and decay below 0.1 in ~10 steps; an instruction-prompted V is unlikely to sit in the same subspace as a real training V at all. A "clever hint" version (prompts that describe the grading environment without instructing the hack — analogous to Aria's inoculation prompts) would be defensible because the hack would still be model-decided. But this is already covered by the G2 screen: the `inoc-prompt-loophole-s65`, `inoc-prompt-passtests-s65`, and `inoc-prompt-evalenv-s65` checkpoints (#123-#125, regrades #131-#133) ARE base Qwen3-4B trained under hint-priming prompts. Whatever mechanisms survive that training are exactly the "model still decides to hack despite environment hints" set that Plan B would have been trying to construct from scratch. Decision: don't run any prompted-pair generation. If the G2 screen comes back uniformly EC-dominant across all 8 candidates, the project conclusion is "Aria's intervention checkpoints don't give detector-class diversity" and we write that up as a negative result rather than papering over it with synthetic pairs. ## Errors | Task | Error | Resolution | |------|-------|------------|