Unify condition lowering logic across Pattern 2/4 with trait-based API. New infrastructure: - condition_lowering_box.rs: ConditionLoweringBox trait + ConditionContext (293 lines) - ExprLowerer implements ConditionLoweringBox trait (+51 lines) Pattern migrations: - Pattern 2 (loop_with_break_minimal.rs): Use trait API - Pattern 4 (loop_with_continue_minimal.rs): Use trait API Benefits: - Unified condition lowering interface - Extensible for future lowering strategies - Clean API boundary between patterns and lowering logic - Zero code duplication Test results: 911/911 PASS (+2 new tests) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
63 lines
4.2 KiB
Markdown
63 lines
4.2 KiB
Markdown
# Phase 238-EX: ExprLowerer/ScopeManager Scope Boundaries
|
||
|
||
目的: ExprLowerer / ScopeManager / ConditionEnv / LoopBodyLocalEnv / UpdateEnv の責務と参照範囲を文章で固定し、「誰がどこまで見てよいか」を SSOT 化する。コード変更は行わず、ガイドラインと境界ルールを整備するフェーズだよ。
|
||
|
||
---
|
||
|
||
## 1. 対象コンポーネント
|
||
|
||
- ExprLowerer(条件式 lowering 箱)
|
||
- ScopeManager(名前解決の窓口)
|
||
- ConditionEnv(条件式用の名前→ValueId マップ)
|
||
- LoopBodyLocalEnv(body-local init 専用の環境)
|
||
- UpdateEnv(carrier 更新専用の環境)
|
||
|
||
---
|
||
|
||
## 2. 境界ルール(原則)
|
||
|
||
- **名前解決は ScopeManager 経由のみ**: ExprLowerer は ConditionEnv / LoopBodyLocalEnv / CapturedEnv / CarrierInfo に直接触らない。ScopeManager の lookup/scope_of を唯一の窓口とする。
|
||
- **条件式から UpdateEnv へは触らない**: header/break/continue 条件は carrier 更新専用の UpdateEnv にアクセスしない。更新式は UpdateEnv / LoopBodyLocalInitLowerer に限定。
|
||
- **ConditionEnv は「条件で参照する値のみ」**: loop var / carriers / ConditionOnly / captured const など、条件式で参照する JoinIR ValueId のみを持ち、body-local を直接含めない(昇格する場合は ScopeManager+CarrierInfo 経由)。
|
||
- **LoopBodyLocalEnv は init 専用**: body 内 local 定義の init 式だけを扱い、条件式の参照は ScopeManager が仲介する(条件側から直接参照しない)。
|
||
- **Fail-Fast / by-name 禁止**: ExprLowerer/ScopeManager は「名前ヒューリスティック」や silent fallback を行わず、Unsupported/NotFound は明示エラー(上位で Fail-Fast ポリシーに従う)。
|
||
|
||
---
|
||
|
||
## 3. JsonParser/selfhost 条件パターンへの対応(237 カタログとの橋渡し)
|
||
|
||
- **優先対応(YES/PARTIAL で拾いたいもの)**
|
||
- P2/P4 header の単純比較 (`i < n`, `p < s.length()`): LoopParam + OuterLocal/Captured/MethodCall(length)。ScopeManager で loop var / captured const を解決し、MethodCall 対応は将来 ExprLowerer 拡張で吸収。
|
||
- P2 break 条件 `digit_pos < 0`: body-local 昇格済み (ConditionOnly carrier)。ScopeManager が `digit_pos`→`is_digit_pos` を解決。
|
||
- substring + equality の単純分岐(`ch == "]"` など): LoopBodyLocal + MethodCall(substring)。MethodCallLowerer 経由で ExprLowerer へ委譲する拡張が必要。
|
||
- **当面後回し**
|
||
- return ベースの終了(_parse_string のように return で抜ける): LoopPattern 側の設計が先。ExprLowerer は条件式評価に限定。
|
||
- 複合的な文字列操作(escape 処理など): legacy ConditionEnv +専用 lowerer 維持。
|
||
|
||
ScopeManager が解決する名前の例:
|
||
- LoopParam: `i`, `p`, `start`, `end`
|
||
- Captured/OuterLocal: `len`, `n`, `digits`
|
||
- Promoted LoopBodyLocal: `digit_pos` → `is_digit_pos`
|
||
- BodyLocal(条件で参照する場合): `ch`, `temp` など(LoopBodyLocalEnv → ScopeManager 経由)
|
||
|
||
---
|
||
|
||
## 4. LAYER_GUARD / ガード実装案(構造メモ)
|
||
|
||
- 将来的に `src/mir/join_ir/lowering/LAYER_GUARD.rs` 相当で、以下を静的に検知する案を検討:
|
||
- ExprLowerer が ConditionEnv / LoopBodyLocalEnv を直接 import していないか。
|
||
- ScopeManager が UpdateEnv を参照していないか。
|
||
- condition_to_joinir から ScopeManager をバイパスするパスが残っていないか。
|
||
- 既存の AGENTS 原則(by-name ハードコード禁止 / Fail-Fast / 構造優先)を lint 的に守る仕組みの叩き台にする。
|
||
|
||
---
|
||
|
||
## 5. 次フェーズ候補(実装 TODO のメモ)
|
||
|
||
- ExprLowerer に MethodCall(length/substring/indexOf) の条件用 lowering を追加(カタログ JP-01/03/04/06/08/07 対応)。
|
||
- ScopeManager のガード強化(条件式から UpdateEnv へのアクセス禁止を型/モジュール境界で表現)。
|
||
- ConditionEnv 構築を ScopeManager 中心に巻き直す(ConditionEnvBuilder v2 を ScopeManager-front に寄せる)。
|
||
- LAYER_GUARD 的な静的チェックの導入検討。***
|
||
Status: Active
|
||
Scope: ExprLowerer スコープ境界(JoinIR/ExprLowerer ライン)
|