Files
hakorune/docs/development/current/main/phase211-loop-candidate-selection.md

278 lines
11 KiB
Markdown
Raw Normal View History

# Phase 211: JsonParser 次の 1 手(中規模ループ候補選定)
**Phase**: 211
**Date**: 2025-12-09
**Status**: 🎯 設計フェーズ(コード実装なし)
**Prerequisite**: Phase 210 完了(軽量ループ 3 本実戦成功)
---
## 🎯 Phase 211 の目的
Phase 210 で「軽量ループ 3 本」が完全成功したため、次は **「中規模の複雑さを持つループ 1 本」** を選び、既存の Pattern/P5 boxes をどう組み合わせるか **設計のみ** 行う。
### 📋 作業範囲(明確化)
-**やること**: ループ 1 本選定 → Pattern/boxes マッピング → 組み合わせ戦略設計
-**やらないこと**: コード実装、ハーネス作成、テスト実行
- 🎯 **成果物**: Phase 212+ で実装する際の「設計図」
---
## Task 211-1: 中規模ループ候補の選定
### 候補 A: `_parse_string` 簡略版
**元の仕様** (Phase 181 より):
```hako
_parse_string(pos) {
local i = pos
local escaped = 0 // LoopBodyLocal (フラグ)
local buf = new ArrayBox() // Buffer構築
loop(i < len) {
local ch = s.char_at(i)
if ch == quote and escaped == 0 { break } // 終了条件
if ch == backslash {
escaped = 1 // フラグ切り替え
} else {
if escaped == 1 {
buf.append(escape_char(ch)) // エスケープ処理
escaped = 0
} else {
buf.append(ch)
}
}
i = i + 1
}
return buf.to_string()
}
```
**簡略版スコープ** (Phase 211 用):
-`escaped` フラグLoopBodyLocal の if 分岐)
-`buf` バッファ構築ArrayBox.append
-`escape_char()` 詳細処理Phase 211 では省略 → "X" で代用)
- ❌ StringBox.to_string()(単純化のため最終 return は buf のまま)
**複雑さの軸**:
- **A軸 (更新)**: `i = i + 1` Simple+ `escaped` フラグ切り替えIfPHI 必要)
- **B軸 (脱出)**: `break` Pattern 2 Break
- **C軸 (条件)**: `ch == quote and escaped == 0` Multi-condition
- **D軸 (変数)**: `i`, `escaped`, `buf` 3 carriers
### 候補 B: selfhost if-sum パターン
**元の仕様** (Phase 181 より):
```hako
// FuncScannerBox._sum_def_count() の簡略版
_sum_def_count(defs) {
local sum = 0
local i = 0
loop(i < defs.len()) {
local item = defs.get(i)
if item != null {
sum = sum + 1 // 条件付き加算
}
i = i + 1
}
return sum
}
```
**複雑さの軸**:
- **A軸 (更新)**: `sum = sum + 1` (条件内)+ `i = i + 1` (無条件)
- **B軸 (脱出)**: なし(自然終了)
- **C軸 (条件)**: `item != null` Simple
- **D軸 (変数)**: `sum`, `i` 2 carriers
---
## Task 211-2: Pattern/Boxes マッピング(候補ごと)
### 候補 A マッピング: `_parse_string` 簡略版
| 軸 | 要求 | 既存 Pattern/Box | Phase 210 時点の対応状況 |
|---|-----|----------------|----------------------|
| **A軸** | `i = i + 1` + `escaped` フラグ | Pattern 2 + IfPHI | ✅ Phase 210 で multi-carrier 確認済み |
| **B軸** | `break` | Pattern 2 Break | ✅ Phase 210 で動作確認済み |
| **C軸** | `ch == quote and escaped == 0` | ConditionLowerer + Multi-condition | ✅ Phase 169 で `and` 対応済み |
| **D軸** | 3 carriers (`i`, `escaped`, `buf`) | CarrierInfo + Multi-carrier | ✅ Phase 210 で 2-carrier 確認済み3-carrier は未テスト) |
**特殊要素**:
- **LoopBodyLocal**: `escaped` はループ内 if 分岐で更新される「状態フラグ」
- Phase 171 Trim Pattern では「ループ末尾で代入→Carrier 昇格」だったが、今回は **「if 分岐内で更新→PHI 必要」**
- 既存 IfPHI ロジックPhase 61で対応可能か要検証
- **Buffer 構築**: `buf.append(ch)` は BoxCall だが、JoinIR では BoxCall は Opaque 扱い
- Phase 210 で BoxCall 自体は問題なし(既存パターンで動作)
**Phase 211 での設計焦点**:
1. `escaped` フラグを Carrier として扱うか、LoopBodyLocal+IfPHI で扱うか
2. 3-carrier (i, escaped, buf) の PHI 配線が既存ロジックで通るか
### 候補 B マッピング: selfhost if-sum パターン
| 軸 | 要求 | 既存 Pattern/Box | Phase 210 時点の対応状況 |
|---|-----|----------------|----------------------|
| **A軸** | `sum = sum + 1` (条件内) + `i = i + 1` | Pattern 1 + IfPHI | ✅ IfPHI は Phase 61 で実装済み |
| **B軸** | なし(自然終了) | Pattern 1 Simple | ✅ Phase 210 で確認済み |
| **C軸** | `item != null` | ConditionLowerer | ✅ 比較演算子対応済み |
| **D軸** | 2 carriers (`sum`, `i`) | CarrierInfo | ✅ Phase 210 で動作確認済み |
**特殊要素**:
- **条件付き更新**: `sum = sum + 1` が if ブロック内
- Phase 61 IfPHI で対応可能(ループ内 if は Merge 経由で Carrier に PHI 接続)
**Phase 211 での設計焦点**:
1. ループ内 if の `sum` 更新が IfPHI → Loop Header PHI に正しく接続されるか確認
---
## Task 211-3: 推奨候補の選定と組み合わせ戦略
### 🎯 推奨: 候補 B (`selfhost if-sum`) を Phase 211 で選定
**理由**:
1. **既存 boxes で完全カバー可能**
- Pattern 1 Simple + IfPHI + Multi-carrierすべて Phase 210 で動作確認済み)
- 新規要素: 「ループ内 if の条件付き更新」のみ
2. **検証価値が高い**
- Phase 61 IfPHI が「ループ内 if」でも正しく動作するか実戦確認
- selfhost 実用パターン(`_sum_def_count` 等)の代表例
3. **Phase 212 実装が軽量**
- ハーネス作成が簡単ArrayBox.get + null チェック)
- デバッグが容易(条件分岐 1 箇所のみ)
**候補 A を Phase 212 以降に回す理由**:
- 3-carrier は Phase 210 で未テスト2-carrier までしか確認していない)
- `escaped` フラグの LoopBodyLocal+IfPHI 処理が複雑
- Phase 211 で「ループ内 if 更新」を先に確認してから、Phase 212+ で 3-carrier に進む方が安全
---
## Task 211-4: Boxes 組み合わせ設計(候補 B: if-sum
### 使用する既存 Boxes
| Box 名 | 役割 | Phase 210 確認状況 |
|-------|-----|------------------|
| **LoopPatternRouter** | Pattern 1 ルーティング | ✅ Phase 210 で動作確認 |
| **SimpleWhileMinimal** | Pattern 1 lowering | ✅ Phase 210 で動作確認 |
| **ConditionLowerer** | `item != null` → JoinIR | ✅ Phase 169/210 で確認 |
| **CarrierInfo** | `sum`, `i` の metadata 管理 | ✅ Phase 210 で確認 |
| **IfPhiContext** | ループ内 if の PHI 生成 | ⚠️ Phase 61 実装済みだが、ループ内 if での実戦は未確認 |
| **JoinValueSpace** | ValueId 割り当て | ✅ Phase 210 で region 分離確認 |
### 処理フロー設計Phase 212 実装時の想定)
```
1. LoopPatternRouter が Pattern 1 を検出
2. SimpleWhileMinimal が呼び出される
3. CarrierInfo が `sum`, `i` を carrier として登録
4. Loop Header PHI 生成:
- PHI(sum): entry=0, back_edge=sum_updated
- PHI(i): entry=0, back_edge=i_updated
5. ConditionLowerer が `i < defs.len()` を JoinIR に変換
6. ループ本体:
- `local item = defs.get(i)` → JoinIR BoxCall (Opaque)
- `if item != null { ... }` → IfPhiContext 起動
6a. IfPhiContext が if ブロック内の `sum = sum + 1` を処理
- then ブロック: sum_updated = sum_current + 1
- else ブロック: sum_updated = sum_current (変更なし)
- Merge 点: PHI(sum_updated) ← [then: sum+1, else: sum]
- `i = i + 1` → 無条件更新
7. Loop Back Edge:
- sum_updated → Header PHI(sum) の back_edge
- i_updated → Header PHI(i) の back_edge
8. Exit PHI:
- PHI(sum_final): loop_exit ← Header PHI(sum)
- PHI(i_final): loop_exit ← Header PHI(i)
```
### 重要な設計ポイント
**IfPhiContext の責務**:
- ループ内 if の **Merge 点で PHI 生成** → この PHI が Loop Header PHI の back_edge に接続される
- Phase 61 実装時は「ループ外 if」を想定していたが、**ループ内 if でも同じロジックが適用できる** はず
**検証ポイントPhase 212 で確認)**:
1. IfPhiContext がループ内 if を正しく検出するか
2. Merge PHI が Header PHI の back_edge に正しく接続されるか
3. `sum` の ValueId が Param region (100-999) に割り当てられるかPhase 201/205 要件)
---
## Task 211-5: Phase 212+ 実装スコープ定義
### Phase 212: if-sum ハーネス実装・実行
**スコープ**:
-`apps/tests/phase212_if_sum_min.hako` 作成
- ✅ 実行 → 観測Phase 210 と同じ Fail-Fast 戦略)
- ✅ IfPhiContext のループ内 if 動作確認
- ✅ phase212-if-sum-observation.md にログ記録
**期待される成果**:
- ループ内 if の条件付き更新が正しく動作
- IfPhiContext → Header PHI 接続が正常
-**「ループ内 if + multi-carrier」パターンが実戦確認済み** になる
### Phase 213+: 段階的拡張(候補 A 等)
**Phase 213**: 3-carrier テスト(`_parse_string` 簡略版の前段階)
- 候補: `i`, `sum`, `count` の 3-carrier ループ(ダミー処理)
- 目的: 3-carrier の PHI 配線が既存ロジックで通るか確認
**Phase 214**: `_parse_string` 簡略版(`escaped` フラグ + `buf` バッファ)
- 候補 A の実装
- 条件: Phase 213 で 3-carrier が成功していること
**Phase 215+**: 残りの JsonParser ループPhase 181 inventory より)
- `_read_array`, `_read_object` 等の再帰呼び出しパターン
- `_parse_hex` 等の特殊処理
---
## 📊 Phase 211 の成果物(このドキュメント)
### ✅ 達成したこと
1. **候補選定**: 候補 B (`selfhost if-sum`) を Phase 212 実装対象に選定
2. **Pattern/Boxes マッピング**: 既存 boxes で完全カバー可能と確認
3. **組み合わせ戦略**: IfPhiContext → Header PHI 接続フローを設計
4. **Phase 212+ スコープ**: 段階的拡張計画を定義
### 🎯 Phase 212 への引き継ぎ事項
- **実装対象**: `apps/tests/phase212_if_sum_min.hako`(条件付き加算ループ)
- **検証ポイント**: IfPhiContext のループ内 if 動作、Header PHI 接続
- **期待結果**: Phase 210 同様の完全成功Fail-Fast トリガーなし)
---
## 📝 補足: Phase 210 との差分
| 項目 | Phase 210 | Phase 211 |
|-----|----------|----------|
| **複雑さ** | 軽量Pattern 1/2 基本形) | 中規模(ループ内 if 更新) |
| **新規要素** | なし(既存確認のみ) | IfPhiContext のループ内適用 |
| **Carrier 数** | 2 まで確認 | 2Phase 213 で 3 に拡張予定) |
| **アプローチ** | 実戦観測 | 設計のみPhase 212 で実装) |
---
**Phase 211 完了条件**: ✅ このドキュメントの作成完了
**次のステップ**: Phase 212if-sum ハーネス実装・実行)