Files
hakorune/src/parser/declarations/box_def/members/methods.rs
nyash-codex f9d100ce01 chore: Phase 25.1 完了 - LoopForm v2/Stage1 CLI/環境変数削減 + Phase 26-D からの変更
Phase 25.1 完了成果:
-  LoopForm v2 テスト・ドキュメント・コメント完備
  - 4ケース(A/B/C/D)完全テストカバレッジ
  - 最小再現ケース作成(SSAバグ調査用)
  - SSOT文書作成(loopform_ssot.md)
  - 全ソースに [LoopForm] コメントタグ追加

-  Stage-1 CLI デバッグ環境構築
  - stage1_cli.hako 実装
  - stage1_bridge.rs ブリッジ実装
  - デバッグツール作成(stage1_debug.sh/stage1_minimal.sh)
  - アーキテクチャ改善提案文書

-  環境変数削減計画策定
  - 25変数の完全調査・分類
  - 6段階削減ロードマップ(25→5、80%削減)
  - 即時削除可能変数特定(NYASH_CONFIG/NYASH_DEBUG)

Phase 26-D からの累積変更:
- PHI実装改善(ExitPhiBuilder/HeaderPhiBuilder等)
- MIRビルダーリファクタリング
- 型伝播・最適化パス改善
- その他約300ファイルの累積変更

🎯 技術的成果:
- SSAバグ根本原因特定(条件分岐内loop変数変更)
- Region+next_iパターン適用完了(UsingCollectorBox等)
- LoopFormパターン文書化・テスト化完了
- セルフホスティング基盤強化

Co-Authored-By: Claude <noreply@anthropic.com>
Co-Authored-By: ChatGPT <noreply@openai.com>
Co-Authored-By: Task Assistant <task@anthropic.com>
2025-11-21 06:25:17 +09:00

86 lines
3.0 KiB
Rust

//! Methods parsing (name(params){ body }) with special birth() prologue
use crate::ast::{ASTNode, Span};
use crate::parser::common::ParserUtils;
use crate::parser::{NyashParser, ParseError};
use crate::tokenizer::TokenType;
/// Try to parse a method declaration starting at `method_name` (already consumed identifier).
/// Returns Some(method_node) when parsed; None when not applicable (i.e., next token is not '(').
pub(crate) fn try_parse_method(
p: &mut NyashParser,
method_name: String,
is_override: bool,
birth_once_props: &Vec<String>,
) -> Result<Option<ASTNode>, ParseError> {
if !p.match_token(&TokenType::LPAREN) {
return Ok(None);
}
p.advance(); // consume '('
let mut params = Vec::new();
while !p.match_token(&TokenType::RPAREN) && !p.is_at_end() {
crate::must_advance!(p, _unused, "method parameter parsing");
if let TokenType::IDENTIFIER(param) = &p.current_token().token_type {
params.push(param.clone());
p.advance();
}
if p.match_token(&TokenType::COMMA) {
p.advance();
}
}
p.consume(TokenType::RPAREN)?;
let mut body = p.parse_block_statements()?;
// Inject eager init for birth_once at the very beginning of user birth()
if method_name == "birth" && !birth_once_props.is_empty() {
let mut injected: Vec<ASTNode> = Vec::new();
for pprop in birth_once_props.iter() {
let me_node = ASTNode::Me {
span: Span::unknown(),
};
let compute_call = ASTNode::MethodCall {
object: Box::new(me_node.clone()),
method: format!("__compute_birth_{}", pprop),
arguments: vec![],
span: Span::unknown(),
};
let tmp = format!("__ny_birth_{}", pprop);
let local_tmp = ASTNode::Local {
variables: vec![tmp.clone()],
initial_values: vec![Some(Box::new(compute_call))],
span: Span::unknown(),
};
let set_call = ASTNode::MethodCall {
object: Box::new(me_node.clone()),
method: "setField".to_string(),
arguments: vec![
ASTNode::Literal {
value: crate::ast::LiteralValue::String(format!("__birth_{}", pprop)),
span: Span::unknown(),
},
ASTNode::Variable {
name: tmp,
span: Span::unknown(),
},
],
span: Span::unknown(),
};
injected.push(local_tmp);
injected.push(set_call);
}
let mut new_body = injected;
new_body.extend(body.into_iter());
body = new_body;
}
let method = ASTNode::FunctionDeclaration {
name: method_name.clone(),
params,
body,
is_static: false,
is_override,
span: Span::unknown(),
};
Ok(Some(method))
}