# Phase 282 P9a 完了 (Scope-Limited Integration) ## 実装内容 - **common_helpers.rs 作成**: 4グループの共通ヘルパー統合 (316行) - Group 1: Control Flow Counting (count_control_flow - 汎用カウンター) - Group 2: Control Flow Detection (has_break/continue/return_statement) - Group 3: Condition Validation (extract_loop_variable, is_true_literal) - Group 4: Pattern5専用ヘルパー (validate_continue_at_end, validate_break_in_simple_if) - **Pattern統合完了**: Pattern5 → Pattern4 → Pattern2 → Pattern1 - Pattern5: ~90行削減 (5 tests PASS) - Pattern4: ~66行削減 (5 tests PASS) - Pattern2: ~67行削減 (4 tests PASS) - Pattern1: ~28行削減 (3 tests PASS) - Pattern3: 別フェーズに延期(pattern固有ロジック除外) ## 成果 - **コード削減**: ~251行(Pattern3除く、total ~400行見込み) - **テスト**: 40 unit tests PASS (23 common_helpers + 17 extractors) - **スモークテスト**: 45 PASS, 1 pre-existing FAIL(退行ゼロ) - **ビルド警告**: 130 → 120 (-10) ## USER CORRECTIONS適用済み 1. ✅ スコープ限定(共通ロジックのみ、pattern固有除外) 2. ✅ Placeholder禁止(SSOT違反排除) 3. ✅ 統合順序変更(Pattern3を最後/別フェーズへ) ## 追加ドキュメント - Phase 284 計画追加(Return as ExitKind SSOT) - 10-Now.md, 30-Backlog.md 更新 🎯 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
294 lines
18 KiB
Markdown
294 lines
18 KiB
Markdown
# Self Current Task — Backlog (main)
|
||
|
||
Status: Active
|
||
Scope: 「次にやる候補」を短く列挙するメモ。現状は `docs/development/current/main/10-Now.md` を入口にする。
|
||
Related:
|
||
- `docs/development/current/main/10-Now.md`
|
||
- `docs/development/current/main/DOCS_LAYOUT.md`
|
||
|
||
## 直近(JoinIR/selfhost)
|
||
|
||
- **Phase 284(planned, design-first): Return as ExitKind SSOT(patternに散らさない)**
|
||
- 目的: `return` を “pattern最適化の個別実装” にせず、`ExitKind` と `compose::*` / `emit_frag()` に収束させる
|
||
- ねらい: 移行期間中の「二重ルール」「検出の穴」を減らし、将来の pattern 増殖を防ぐ
|
||
- 入口: `docs/development/current/main/phases/phase-284/README.md`
|
||
- P0(docs-only): `docs/development/current/main/phases/phase-284/P0-INSTRUCTIONS.md`
|
||
- SSOT:
|
||
- Composition: `src/mir/builder/control_flow/edgecfg/api/compose.rs`
|
||
- Terminator emission: `emit_frag()`(`src/mir/builder/control_flow/edgecfg/api/emit.rs`)
|
||
- Frag docs: `docs/development/current/main/design/edgecfg-fragments.md`
|
||
- 進め方:
|
||
- P0(docs-only)で “return の意味” と “Ok(None)/Err” の境界を固定
|
||
- P1+ で Rust/LLVM の実装を SSOT に収束(pattern側に例外実装を増やさない)
|
||
|
||
- (✅ done)**Phase 282**: Router shrinkage + detection SSOT + extractor refactor
|
||
- 完了: `docs/development/current/main/phases/phase-282/README.md`
|
||
|
||
(✅ done)**Phase 279 P0**: Type propagation pipeline SSOT 統一(lifecycle / JoinIR / LLVM の二重化解消)
|
||
- 完了: `docs/development/current/main/phases/phase-279/README.md`
|
||
|
||
- **Phase 272(✅ complete): Pattern6/7 を Frag+emit_frag へ吸収(段階適用)**
|
||
- 目的: scan系 loop の CFG 構築を `Frag/ExitKind` 合成へ寄せ、pattern列挙の増殖を止める
|
||
- 完了: P0.1(Pattern6)✅ + P0.2(Pattern7)✅
|
||
- 入口: fixture/smoke を SSOT として固定(Pattern6→Pattern7 の順で段階適用)
|
||
- 詳細: `phases/phase-272/README.md`
|
||
|
||
- (✅ done)**Phase 273**: Plan line SSOT(Pattern6/7)
|
||
- 完了: `docs/development/current/main/phases/phase-273/README.md`
|
||
|
||
- (✅ done)**Phase 280**: Frag composition SSOT positioning
|
||
- 完了: `docs/development/current/main/phases/phase-280/README.md`
|
||
|
||
- (✅ done)**Phase 281**: compose adoption(Pattern6/7, P0–P3)
|
||
- 完了: `docs/development/current/main/phases/phase-281/README.md`
|
||
|
||
- (✅ done)**Phase 277/278**: PHI strict + env var 収束
|
||
- 完了: `docs/development/current/main/phases/phase-277/README.md`
|
||
- 完了: `docs/development/current/main/phases/phase-278/README.md`
|
||
|
||
|
||
- **Phase 274(active, design-first): Type SSOT Alignment(local + dynamic runtime)**
|
||
- 入口SSOT: `docs/reference/language/types.md`
|
||
- P1(✅完了): Rust VM が `TypeOp(Check/Cast)` を実行可能(`is/as` が動く)
|
||
- P2(✅完了): LLVM ライン(llvmlite harness)の `TypeOp` を SSOT に合わせる
|
||
- P3(decision): truthiness / equality / compare / `+` の coercion を SSOT として固定(必要なら “仕様変更フェーズ” を追加)
|
||
- 詳細: `phases/phase-274/README.md`
|
||
|
||
- **Phase 270(✅ 完了): JoinIR-only minimal loop SSOT**
|
||
- `apps/tests/phase270_p0_loop_min_const.hako` + VM smoke で “最小 const loop” を固定(exit=3)
|
||
- Pattern1 は test-only stub のため不適合 → Pattern9(AccumConstLoop)を橋渡しとして追加
|
||
- 詳細: `phases/phase-270/README.md`
|
||
|
||
- **Phase 271(✅ 完了, docs-only): Bridge pattern 撤去条件SSOT**
|
||
- 対象: bridge pattern(例: `Pattern9_AccumConstLoop`)
|
||
- 目的: 「汎用化しない」「Frag 合成へ吸収して削除する」を SSOT 化
|
||
- 成果物:
|
||
- `docs/development/current/main/design/edgecfg-fragments.md` の `Bridge patterns(撤去条件SSOT)` に “bridge contract” テンプレを追加
|
||
- `Pattern9_AccumConstLoop` の撤去条件(fixture/smoke/手順)を同セクションに明文化
|
||
- SSOT: `docs/development/current/main/design/edgecfg-fragments.md`
|
||
|
||
- **Phase 269 P1(✅ 完了): Pattern8 を EdgeCFG で実装(SSA を閉じる)**
|
||
- 方針: emission 入口で Frag 構築(break/continue 無しなので `compose::loop_()` は使わず手配線)
|
||
- 完了: header に `i` の PHI を追加して SSA を閉じた(`i_current = phi [i_init, preheader], [i_next, step]`)
|
||
- early-exit の `return false` は Return wire、`return true` は loop 後 AST に任せる
|
||
- Pattern8 の返り値は当面 `void`(loop-statement 扱い)
|
||
- 補足(DONE): static box の `this/me` は MethodCall 共通入口で static call に正規化済み(Pattern8 は現状 static box 文脈を対象外)
|
||
- 詳細: `phases/phase-269/README.md`
|
||
|
||
- **Phase 270+(planned): Pattern6/7 への Frag 適用**
|
||
- Pattern6: Match scan(index_of 系)
|
||
- Pattern7: Split scan(split 系)
|
||
- 統一的なループ処理パターン確立
|
||
- Pattern 分岐削減
|
||
|
||
- **収束方針(SSOT案): Expr/Condition/Control の 3 箱分割**
|
||
- ExprLowererBox(式SSOT): `AST(expr)` → `(prelude, value)`(ANF含む)。pure/impure/whitelist/strict を集約(入口SSOT)。
|
||
- ConditionLowererBox(条件→分岐SSOT): `AST(cond)` → `BranchPlan`。評価順は ExprLowererBox に委譲し、`&&/||` は制御語彙で扱う。
|
||
- ControlLowererBox(制御SSOT): `StepNode/ControlTree` → JoinIR(継続 + env)。`if/loop` を担当し、条件は ConditionLowererBox に委譲。
|
||
|
||
- **Phase 141 P2+(planned): Call/MethodCall(effects + typing を分離して段階投入)**
|
||
- ねらい: pure/impure 境界を壊さずに、impure lowering を段階投入する。
|
||
- 前提(DONE):
|
||
- Phase 141 P1.5: known intrinsic allowlist + available_inputs 3-source merge + diagnostics
|
||
- 受け入れ条件:
|
||
- out-of-scope は `Ok(None)` でフォールバック(既定挙動不変)
|
||
- effects の順序付けは SSOT で固定してから解禁(by-name 増殖禁止)
|
||
|
||
- **Phase 144-anf(planned): impure 式導入の順序固定(ANF)**
|
||
- ねらい: `x + f(y)` 等の “pure + impure 混在” で評価順が仕様になる前に、ANF で順序固定を SSOT 化する
|
||
- 入口: `docs/development/current/main/phases/phase-144-anf/INSTRUCTIONS.md`
|
||
- 受け入れ条件:
|
||
- impure を lowering できない場合は `Ok(None)` でフォールバック(既定挙動不変)
|
||
- dev/strict では「順序固定の欠落」を Fail-Fast(診断に順序ログを含める)
|
||
|
||
- **Phase 143-loopvocab R0(planned): Contract SSOT 抽出(refactor P0 → modular components)**
|
||
- 目的: loop_true_if_break_continue.rs を「検出/契約/変換」に分割し、P1/P2 での if分岐増殖を防ぐ
|
||
- 実装:
|
||
- 新ファイル: `src/mir/control_tree/normalized_shadow/common/loop_if_exit_contract.rs`
|
||
- `enum LoopIfExitThen { Break, Continue }`
|
||
- `struct LoopIfExitShape { has_else: bool, then: LoopIfExitThen, else_: Option<LoopIfExitThen>, cond_scope: ExprLoweringScope }`
|
||
- `enum OutOfScopeReason { NotLoopTrue, BodyNotSingleIf, ThenNotExit, ElseNotSupported, CondOutOfScope(...) }`
|
||
- Refactor: loop_true_if_break_continue.rs は「shape抽出 → lower」だけに縮退(SSOT は contract側)
|
||
- Tests: unit test を dedicated module へ分離(test maintainability)
|
||
- 受け入れ条件:
|
||
- cargo check ✅(no errors)
|
||
- P1/P2 での if分岐を防ぐ(contract で決定性を保証)
|
||
- out-of-scope は `Ok(None)` で一貫(既定挙動不変)
|
||
|
||
- **Phase 143-loopvocab P1(planned): continue 語彙追加**
|
||
- 対象: `loop(true) { if(cond_pure) continue }` を same lowering に通す
|
||
- 実装:
|
||
- LoopIfExitShape で `LoopIfExitThen::Continue` を許可
|
||
- JoinModule: if true → loop_step (continue semantics)
|
||
- Fixtures: `phase143_loop_true_if_continue_min.hako`
|
||
- Smoke: VM + LLVM EXE
|
||
- Out-of-scope は `Ok(None)` のまま
|
||
|
||
(DONE)Phase 143-loopvocab P2: else 対称化(B-C / C-B)
|
||
- 記録: `docs/development/current/main/10-Now.md`
|
||
|
||
- **Phase 143-loopvocab P3+(planned): impure conditions 対応**
|
||
- 目的: `if(cond_impure) break/continue` を ANF/順序固定の上で段階投入する
|
||
- 方針: Phase 145-anf の契約(hoist + left-to-right)を条件式にも適用
|
||
|
||
- **Phase 263+(planned): Pattern2 LoopBodyLocal promotion(seg)**
|
||
- 目的: Stage‑B compile(bundle_resolver系)で露出している Pattern2 `LoopBodyLocal(seg)` を受理し、quick の first FAIL を進める
|
||
- 受け入れ条件:
|
||
- 最小再現 fixture + smoke で固定(先に失敗を SSOT 化)
|
||
- Pattern2 が不成立のときは “部分続行” せず `Ok(None)` で fallback(既定挙動不変)
|
||
|
||
- **(DONE)Phase 263 P0.2: Pattern2 PromoteDecision API hardening**
|
||
- 入口SSOT: `src/mir/builder/control_flow/joinir/patterns/pattern2/api/`
|
||
- `PromoteDecision::{Promoted, NotApplicable, Freeze}` と `try_promote(...)` に参照点を収束(Option揺れを撤去)
|
||
|
||
- **Phase 264(✅ 入口作成完了): EdgeCFG Fragment 入口作成(design-first)**
|
||
- **ステータス**: ✅ 入口作成完了(適用は次フェーズ)
|
||
- **実装内容**:
|
||
- `edgecfg/api/` フォルダに SSOT 入口作成
|
||
- `ExitKind`, `EdgeStub`, `Frag` の型定義
|
||
- `seq`, `if_`, `loop_`, `cleanup` のシグネチャ固定(pub(crate))
|
||
- 最小ユニットテスト 3個
|
||
- ドキュメント連動(edgecfg-fragments.md)
|
||
- **制約遵守**:
|
||
- 既存 pattern6/7/8 未改変
|
||
- merge/EdgeCFG 未改変
|
||
- cargo test -p nyash-rust --lib --no-run 成功確認
|
||
- **次フェーズへの橋渡し**:
|
||
- Phase 265 で Pattern8 適用時に `compose::loop_` を実装
|
||
- 再利用確認後、pattern番号分岐を段階的に削減
|
||
|
||
- **Phase 265 P0(✅ 完了): compose/verify 最小実装**
|
||
- **目的**: 入口SSOTの形を固める(迷子防止)
|
||
- **実装**:
|
||
- compose::loop_() 最小実装(exit集合分類のみ、配線なし)
|
||
- verify_frag_invariants() 最小実装(デバッグガード付き)
|
||
- compose::loop_() ユニットテスト 2個追加
|
||
- **制約**:
|
||
- Pattern8 未改変(P0では触らない、偽Frag回避)
|
||
- 配線ロジックは P1 以降
|
||
- **次**: Phase 265 P1 で配線ロジック + Pattern8適用
|
||
|
||
- **Phase 265 P1(✅ 完了): compose 配線ロジック実装**
|
||
- **目的**: Frag/ExitKind の配線能力を BasicBlockId 層で証明
|
||
- **実装**:
|
||
- EdgeStub.target 追加(Option<BasicBlockId>)
|
||
- compose::loop_() 配線ロジック(Continue → header, Break → after)
|
||
- verify_frag_invariants() 配線契約検証
|
||
- test-only PoC(5個のテスト: 既存2個更新 + 新規3個追加)
|
||
- **配線契約**:
|
||
- Continue(loop_id) の EdgeStub.target = Some(header)
|
||
- Break(loop_id) の EdgeStub.target = Some(after)
|
||
- Normal/Return/Unwind の EdgeStub.target = None(上位へ伝搬)
|
||
- **制約**:
|
||
- MIR 命令生成なし(Frag 層のみ)
|
||
- NormalizedShadow 未適用(Phase 267 に繰り越し)
|
||
|
||
- **(✅ 完了)Phase 265 P2: seq/if_ 実装(wires/exits 分離)**
|
||
- **目的**: 「解決済み配線(wires)」と「未解決 exit(exits)」を分離し、Frag 合成の基本パターンを完成
|
||
- **完了内容**:
|
||
- Frag に `wires: Vec<EdgeStub>` 追加
|
||
- wires/exits 分離設計確立(exits = target None, wires = target Some)
|
||
- loop_() を wires 対応に更新
|
||
- seq(a, b) 実装(a.Normal → wires)
|
||
- if_(header, cond, t, e, join_frag) 実装(t/e.Normal → wires)
|
||
- verify 強化(wires/exits 分離契約、警告のみ)
|
||
- 全テスト PASS(13個: frag 3 + compose 9 + verify 1)
|
||
- **設計判断**:
|
||
- wires/exits 分離で再配線バグ防止
|
||
- if_ は join_frag: Frag で「join 以降」を明確化
|
||
- verify は警告のみ(Err 化は Phase 266 の strict 版で段階導入)
|
||
- **次**: Phase 266 で MIR 命令生成 PoC(emit_wires)、Phase 267 で NormalizedShadow/JoinIR へ実適用
|
||
|
||
- **(✅ 完了)Phase 266: wires → MIR terminator 生成(最小 PoC)**
|
||
- **目的**: wires を MIR terminator に変換する最小 PoC を実装し、Phase 267 での本格適用に備える
|
||
- **完了内容**:
|
||
- emit.rs 作成(emit_wires 実装 + unit test 4個)
|
||
- verify_frag_invariants_strict() 追加(段階導入を壊さない)
|
||
- Jump/Return 対応(Branch は Phase 267)
|
||
- mod.rs 更新(emit module エクスポート)
|
||
- 全テスト PASS(1392 passed: 既存 1388 + 新規 4個)
|
||
- **核心原則**:
|
||
- from ごとにグループ化して1本だけ許可(1 block = 1 terminator 制約)
|
||
- Return は target=None を許可(意味を持たない)
|
||
- verify_frag_invariants() は変更なし(警告のまま、strict 版を別名で用意)
|
||
- Phase 260 terminator 語彙ルールを厳守
|
||
- **設計判断**:
|
||
- from グループ化で terminator 上書きバグ防止
|
||
- Return は target 不要(呼び出し元に戻る)
|
||
- verify strict 版で段階導入を壊さない
|
||
- **次**: Phase 267 で JoinIR Pattern への適用
|
||
|
||
- **(✅ P0 完了)Phase 267: BranchStub + emit_frag(Branch の第一級化)**
|
||
- **目的**: Frag に Branch を第一級で追加し、wires(Jump/Return)と同様に MIR terminator へ落とす入口を作る
|
||
- **完了内容(P0)**:
|
||
- `BranchStub` 追加 + `Frag.branches` 追加
|
||
- `compose::if_` が header→then/else の BranchStub を生成
|
||
- `emit_frag(function, frag)` を追加(`emit_wires` + `set_branch_with_edge_args`)
|
||
- 1 block = 1 terminator(wire/branch の衝突)を Fail-Fast
|
||
- unit tests + `cargo test -p nyash-rust --lib` PASS
|
||
- **P1(延期)**:
|
||
- "層を跨がない実適用"は候補が抽象化層へ委譲済みのため、Phase 268 で体系的に適用する方針
|
||
- **詳細**: `docs/development/current/main/phases/phase-267/README.md`
|
||
|
||
- **(✅ 完了)Phase 268: if_form.rs への Frag 適用 + entry edge-args SSOT化**
|
||
- **目的**: EdgeCFG Fragment を層を跨がずに実戦投入し、compose::if_() の edge-args を SSOT 化
|
||
- **完了内容(P0)**:
|
||
- emission/branch.rs に emit_conditional_edgecfg() 追加(薄いラッパー)
|
||
- if_form.rs を Frag+emit_frag 経由に変更(emit_conditional + emit_jump を削除)
|
||
- emission 層経由で層が綺麗に保たれる
|
||
- **完了内容(P1)**:
|
||
- compose::if_() シグネチャ変更(then_entry_args, else_entry_args 追加)
|
||
- emission/branch.rs から空 EdgeArgs を渡す
|
||
- EdgeCFG テスト更新(compose.rs 2箇所、emit.rs 1箇所)
|
||
- TODO コメント削除完了(Phase 267 P2+ TODO 解消)
|
||
- **テスト結果**:
|
||
- cargo build --release: 成功
|
||
- cargo test --lib --release: 1444/1444 PASS
|
||
- quick smoke: 45/46 PASS(既存状態維持)
|
||
- **核心原則**:
|
||
- emission 層経由で Frag 構築を MirBuilder 層から分離
|
||
- SSOT 原則: compose::if_() は edge-args を内部生成しない
|
||
- **次**: Phase 269 で Pattern6/7/8 への Frag 適用 + fixture/smoke test
|
||
- **詳細**: `docs/development/current/main/phases/phase-268/README.md`
|
||
|
||
- **real-app loop regression の横展開(VM + LLVM EXE)**
|
||
- ねらい: 実コード由来ループを 1 本ずつ最小抽出して fixture/smoke で固定する(段階投入)。
|
||
- 現状: Phase 107(find_balanced_array/object / json_cur 由来)まで固定済み。
|
||
- 次候補: JsonLoader/JsonCur から 1 本ずつ(fixture + integration smoke)で増やす。
|
||
|
||
- **P5b “完全E2E”**(escape skip の実ループを end-to-end で固定)
|
||
- 現状: Phase 94 で VM E2E まで固定済み。次は selfhost 実コード(`apps/selfhost-vm/json_loader.hako`)へ横展開して回帰を減らす。
|
||
- 入口: `docs/development/current/main/phases/phase-94/README.md`
|
||
|
||
- **制御の再帰合成(docs-only → dev-only段階投入)**
|
||
- ねらい: `loop/if` ネストの "構造" を SSOT(ControlTree/StepTree)で表せるようにする
|
||
- 注意: canonicalizer は観測/構造SSOTまで(ValueId/PHI配線は Normalized 側へ)
|
||
- 現状: Phase 119–128(if-only Normalized: reads/inputs/unknown-read/partial-assign keep/merge)まで完了
|
||
- ✅ 完了: Phase 129-C(post-if を post_k continuation で表現)
|
||
- 入口: `docs/development/current/main/design/control-tree.md`
|
||
|
||
## 中期(ループ在庫の残り)
|
||
|
||
- **P5(guard-bounded)**: 大型ループを “小粒度” に割ってから取り込む(分割 or 新契約)
|
||
- **P6(nested loops)**: capability guard で Fail-Fast 維持しつつ、解禁時の契約を先に固定
|
||
|
||
## 中期(制御の表現力)
|
||
|
||
北極星: `docs/development/current/main/design/join-explicit-cfg-construction.md`
|
||
設計メモ: `docs/development/current/main/design/exception-cleanup-async.md`
|
||
|
||
- **catch/cleanup(Invoke)**
|
||
- 追加語彙を `Invoke(ok_edge, err_edge)` に絞って例外 edge を明示する(例外値は edge-args で運ぶ)。
|
||
- 実装タイミング: Phase 260(edge-args terminator 収束)の P1〜P2 以降が推奨。
|
||
- **cleanup/defer(cleanup normalizer)**
|
||
- Return/Throw/Break/Continue を cleanup に寄せる “脱出 edge 正規化” を箱化する(finally の後継としての cleanup)。
|
||
- 実装タイミング: catch/cleanup の次(例外 edge も含めて正規化するため)。
|
||
- **async/await(state machine lowering)**
|
||
- CFG語彙に混ぜず、AsyncLowerBox で state machine 化してから MIR に落とす。
|
||
- 実装タイミング: finally/defer の後(cancel/drop と cleanup の接続を先に固める)。
|
||
|
||
## ドキュメント運用
|
||
|
||
- 重複が出たら「設計 SSOT(design)」に集約し、Phaseログ(phases)は “何をやったか/検証したか” に限定する
|
||
- 調査ログ(investigations)は結論を SSOT に反映してから Historical 化する
|