2025-11-13 16:40:58 +09:00
|
|
|
|
use super::{MirBuilder, ValueId};
|
2025-11-26 10:17:37 +09:00
|
|
|
|
use crate::ast::ASTNode;
|
2025-11-29 15:32:40 +09:00
|
|
|
|
use crate::mir::control_form::IfShape;
|
2025-11-21 06:25:17 +09:00
|
|
|
|
use crate::mir::loop_api::LoopBuilderApi; // for current_block()
|
2025-11-29 15:32:40 +09:00
|
|
|
|
use crate::mir::phi_core::phi_builder_box::PhiBuilderOps;
|
|
|
|
|
|
use crate::mir::{BasicBlockId, ConstValue, MirInstruction};
|
|
|
|
|
|
|
|
|
|
|
|
/// Phase 61-4-F: MirBuilder 用 PhiBuilderOps 実装
|
|
|
|
|
|
///
|
|
|
|
|
|
/// ループ外 if の JoinIR 経路で emit_toplevel_phis() を呼ぶためのラッパー。
|
|
|
|
|
|
struct ToplevelOps<'a>(&'a mut MirBuilder);
|
|
|
|
|
|
|
|
|
|
|
|
impl<'a> PhiBuilderOps for ToplevelOps<'a> {
|
|
|
|
|
|
fn new_value(&mut self) -> ValueId {
|
|
|
|
|
|
self.0.next_value_id()
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn emit_phi(
|
|
|
|
|
|
&mut self,
|
|
|
|
|
|
block: BasicBlockId,
|
|
|
|
|
|
dst: ValueId,
|
|
|
|
|
|
inputs: Vec<(BasicBlockId, ValueId)>,
|
|
|
|
|
|
) -> Result<(), String> {
|
|
|
|
|
|
// merge ブロックの先頭に PHI を挿入
|
2025-11-30 09:38:28 +09:00
|
|
|
|
if let (Some(func), Some(_cur_bb)) =
|
|
|
|
|
|
(self.0.current_function.as_mut(), self.0.current_block)
|
2025-11-29 15:32:40 +09:00
|
|
|
|
{
|
|
|
|
|
|
crate::mir::ssot::cf_common::insert_phi_at_head_spanned(
|
|
|
|
|
|
func,
|
|
|
|
|
|
block,
|
|
|
|
|
|
dst,
|
|
|
|
|
|
inputs,
|
2025-12-15 22:44:38 +09:00
|
|
|
|
self.0.metadata_ctx.current_span(),
|
2025-11-29 15:32:40 +09:00
|
|
|
|
);
|
|
|
|
|
|
} else {
|
2025-11-30 09:38:28 +09:00
|
|
|
|
self.0.emit_instruction(MirInstruction::Phi {
|
|
|
|
|
|
dst,
|
|
|
|
|
|
inputs,
|
|
|
|
|
|
type_hint: None,
|
|
|
|
|
|
})?;
|
2025-11-29 15:32:40 +09:00
|
|
|
|
}
|
|
|
|
|
|
Ok(())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn update_var(&mut self, name: String, value: ValueId) {
|
|
|
|
|
|
self.0.variable_map.insert(name, value);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn get_block_predecessors(&self, block: BasicBlockId) -> Vec<BasicBlockId> {
|
|
|
|
|
|
if let Some(ref func) = self.0.current_function {
|
|
|
|
|
|
func.blocks
|
|
|
|
|
|
.get(&block)
|
|
|
|
|
|
.map(|bb| bb.predecessors.iter().copied().collect())
|
|
|
|
|
|
.unwrap_or_default()
|
|
|
|
|
|
} else {
|
|
|
|
|
|
Vec::new()
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn emit_void(&mut self) -> ValueId {
|
|
|
|
|
|
let void_id = self.0.next_value_id();
|
|
|
|
|
|
let _ = self.0.emit_instruction(MirInstruction::Const {
|
|
|
|
|
|
dst: void_id,
|
|
|
|
|
|
value: ConstValue::Void,
|
|
|
|
|
|
});
|
|
|
|
|
|
void_id
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn set_current_block(&mut self, block: BasicBlockId) -> Result<(), String> {
|
|
|
|
|
|
self.0.current_block = Some(block);
|
|
|
|
|
|
Ok(())
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn block_exists(&self, block: BasicBlockId) -> bool {
|
|
|
|
|
|
if let Some(ref func) = self.0.current_function {
|
|
|
|
|
|
func.blocks.contains_key(&block)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
impl MirBuilder {
|
|
|
|
|
|
/// Lower an if/else using a structured IfForm (header→then/else→merge).
|
|
|
|
|
|
/// PHI-off: edge-copy only on predecessors; PHI-on: Phi at merge.
|
|
|
|
|
|
pub(super) fn lower_if_form(
|
|
|
|
|
|
&mut self,
|
|
|
|
|
|
condition: ASTNode,
|
|
|
|
|
|
then_branch: ASTNode,
|
|
|
|
|
|
else_branch: Option<ASTNode>,
|
|
|
|
|
|
) -> Result<ValueId, String> {
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Reserve a deterministic join id for debug region labeling
|
|
|
|
|
|
let join_id = self.debug_next_join_id();
|
2025-11-26 10:17:37 +09:00
|
|
|
|
// Pre-pin heuristic was deprecated; keep operands as-is for predictability.
|
2025-09-26 04:17:56 +09:00
|
|
|
|
|
2025-09-19 02:07:38 +09:00
|
|
|
|
let condition_val = self.build_expression(condition)?;
|
2025-09-28 20:38:09 +09:00
|
|
|
|
let condition_val = self.local_cond(condition_val);
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
// Create blocks
|
2025-12-15 20:10:36 +09:00
|
|
|
|
let then_block = self.next_block_id();
|
|
|
|
|
|
let else_block = self.next_block_id();
|
|
|
|
|
|
let merge_block = self.next_block_id();
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
// Branch
|
2025-09-26 05:28:20 +09:00
|
|
|
|
let pre_branch_bb = self.current_block()?;
|
2025-09-28 20:38:09 +09:00
|
|
|
|
let mut condition_val = condition_val;
|
|
|
|
|
|
crate::mir::builder::ssa::local::finalize_branch_cond(self, &mut condition_val);
|
2025-11-21 06:25:17 +09:00
|
|
|
|
crate::mir::builder::emission::branch::emit_conditional(
|
|
|
|
|
|
self,
|
|
|
|
|
|
condition_val,
|
|
|
|
|
|
then_block,
|
|
|
|
|
|
else_block,
|
|
|
|
|
|
)?;
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
// Snapshot variables before entering branches
|
|
|
|
|
|
let pre_if_var_map = self.variable_map.clone();
|
|
|
|
|
|
|
2025-09-26 14:34:42 +09:00
|
|
|
|
let trace_if = std::env::var("NYASH_IF_TRACE").ok().as_deref() == Some("1");
|
|
|
|
|
|
|
2025-09-19 02:07:38 +09:00
|
|
|
|
// then
|
2025-09-26 05:28:20 +09:00
|
|
|
|
self.start_new_block(then_block)?;
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Debug region: join then-branch
|
|
|
|
|
|
self.debug_push_region(format!("join#{}", join_id) + "/then");
|
2025-09-21 08:53:00 +09:00
|
|
|
|
// Scope enter for then-branch
|
|
|
|
|
|
self.hint_scope_enter(0);
|
2025-09-19 02:07:38 +09:00
|
|
|
|
let then_ast_for_analysis = then_branch.clone();
|
|
|
|
|
|
self.variable_map = pre_if_var_map.clone();
|
2025-09-26 05:28:20 +09:00
|
|
|
|
// Materialize all variables at block entry via single-pred Phi (correctness-first)
|
|
|
|
|
|
for (name, &pre_v) in pre_if_var_map.iter() {
|
2025-11-06 23:57:24 +09:00
|
|
|
|
let phi_val = self.insert_phi_single(pre_branch_bb, pre_v)?;
|
2025-09-26 05:28:20 +09:00
|
|
|
|
self.variable_map.insert(name.clone(), phi_val);
|
2025-09-26 14:34:42 +09:00
|
|
|
|
if trace_if {
|
|
|
|
|
|
eprintln!(
|
|
|
|
|
|
"[if-trace] then-entry phi var={} pre={:?} -> dst={:?}",
|
|
|
|
|
|
name, pre_v, phi_val
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2025-09-26 05:28:20 +09:00
|
|
|
|
}
|
2025-09-19 02:07:38 +09:00
|
|
|
|
let then_value_raw = self.build_expression(then_branch)?;
|
|
|
|
|
|
let then_exit_block = self.current_block()?;
|
2025-11-01 13:46:34 +09:00
|
|
|
|
let then_reaches_merge = !self.is_current_block_terminated();
|
2025-09-19 02:07:38 +09:00
|
|
|
|
let then_var_map_end = self.variable_map.clone();
|
2025-11-01 13:46:34 +09:00
|
|
|
|
if then_reaches_merge {
|
2025-09-21 08:53:00 +09:00
|
|
|
|
// Scope leave for then-branch
|
|
|
|
|
|
self.hint_scope_leave(0);
|
2025-09-28 20:38:09 +09:00
|
|
|
|
crate::mir::builder::emission::branch::emit_jump(self, merge_block)?;
|
2025-09-19 02:07:38 +09:00
|
|
|
|
}
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Pop then-branch debug region
|
|
|
|
|
|
self.debug_pop_region();
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
// else
|
2025-09-26 05:28:20 +09:00
|
|
|
|
self.start_new_block(else_block)?;
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Debug region: join else-branch
|
|
|
|
|
|
self.debug_push_region(format!("join#{}", join_id) + "/else");
|
2025-09-21 08:53:00 +09:00
|
|
|
|
// Scope enter for else-branch
|
|
|
|
|
|
self.hint_scope_enter(0);
|
2025-11-21 06:25:17 +09:00
|
|
|
|
let (else_value_raw, else_ast_for_analysis, else_var_map_end_opt) =
|
|
|
|
|
|
if let Some(else_ast) = else_branch {
|
|
|
|
|
|
// Reset variable_map BEFORE materializing PHI nodes (same pattern as then-branch)
|
|
|
|
|
|
self.variable_map = pre_if_var_map.clone();
|
|
|
|
|
|
// Materialize all variables at block entry via single-pred Phi (correctness-first)
|
|
|
|
|
|
for (name, &pre_v) in pre_if_var_map.iter() {
|
|
|
|
|
|
let phi_val = self.insert_phi_single(pre_branch_bb, pre_v)?;
|
|
|
|
|
|
self.variable_map.insert(name.clone(), phi_val);
|
|
|
|
|
|
if trace_if {
|
|
|
|
|
|
eprintln!(
|
|
|
|
|
|
"[if-trace] else-entry phi var={} pre={:?} -> dst={:?}",
|
|
|
|
|
|
name, pre_v, phi_val
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2025-11-13 20:16:20 +09:00
|
|
|
|
}
|
2025-11-21 06:25:17 +09:00
|
|
|
|
let val = self.build_expression(else_ast.clone())?;
|
|
|
|
|
|
(val, Some(else_ast), Some(self.variable_map.clone()))
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// No else branch: materialize PHI nodes for the empty else block
|
|
|
|
|
|
self.variable_map = pre_if_var_map.clone();
|
|
|
|
|
|
for (name, &pre_v) in pre_if_var_map.iter() {
|
|
|
|
|
|
let phi_val = self.insert_phi_single(pre_branch_bb, pre_v)?;
|
|
|
|
|
|
self.variable_map.insert(name.clone(), phi_val);
|
|
|
|
|
|
if trace_if {
|
|
|
|
|
|
eprintln!(
|
|
|
|
|
|
"[if-trace] else-entry phi var={} pre={:?} -> dst={:?}",
|
|
|
|
|
|
name, pre_v, phi_val
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2025-11-13 20:16:20 +09:00
|
|
|
|
}
|
2025-11-21 06:25:17 +09:00
|
|
|
|
let void_val = crate::mir::builder::emission::constant::emit_void(self);
|
|
|
|
|
|
// Phase 25.1c/k: Pass PHI-renamed variable_map for empty else branch
|
|
|
|
|
|
// This ensures merge_modified_vars uses correct ValueIds after PHI renaming
|
|
|
|
|
|
(void_val, None, Some(self.variable_map.clone()))
|
|
|
|
|
|
};
|
2025-09-19 02:07:38 +09:00
|
|
|
|
let else_exit_block = self.current_block()?;
|
2025-11-01 13:46:34 +09:00
|
|
|
|
let else_reaches_merge = !self.is_current_block_terminated();
|
|
|
|
|
|
if else_reaches_merge {
|
2025-09-21 08:53:00 +09:00
|
|
|
|
// Scope leave for else-branch
|
|
|
|
|
|
self.hint_scope_leave(0);
|
2025-11-05 18:57:03 +09:00
|
|
|
|
crate::mir::builder::emission::branch::emit_jump(self, merge_block)?;
|
2025-09-19 02:07:38 +09:00
|
|
|
|
}
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Pop else-branch debug region
|
|
|
|
|
|
self.debug_pop_region();
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
// merge: primary result via helper, then delta-based variable merges
|
2025-09-26 05:28:20 +09:00
|
|
|
|
// Ensure PHIs are first in the block by suppressing entry pin copies here
|
|
|
|
|
|
self.suppress_next_entry_pin_copy();
|
|
|
|
|
|
self.start_new_block(merge_block)?;
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Debug region: join merge
|
|
|
|
|
|
self.debug_push_region(format!("join#{}", join_id) + "/join");
|
2025-09-19 02:07:38 +09:00
|
|
|
|
self.push_if_merge(merge_block);
|
|
|
|
|
|
|
2025-11-28 05:01:04 +09:00
|
|
|
|
// Phase 38: Pre-analysis hints removed (JoinIR AST lowering handles assignment detection)
|
|
|
|
|
|
let assigned_then_pre: Option<String> = None;
|
|
|
|
|
|
let assigned_else_pre: Option<String> = None;
|
|
|
|
|
|
let pre_then_var_value: Option<ValueId> = None;
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
2025-11-29 15:08:23 +09:00
|
|
|
|
// Phase 61-4: JoinIR 経路試行(ループ外 If)
|
2025-11-30 14:30:28 +09:00
|
|
|
|
//
|
|
|
|
|
|
// - 実際の有効化判定は config::env 側のポリシー関数に集約する。
|
2025-11-29 15:08:23 +09:00
|
|
|
|
let joinir_enabled = crate::config::env::joinir_if_select_enabled();
|
|
|
|
|
|
let joinir_toplevel = crate::config::env::joinir_if_toplevel_enabled();
|
|
|
|
|
|
let joinir_dryrun = crate::config::env::joinir_if_toplevel_dryrun_enabled();
|
|
|
|
|
|
let mut joinir_success = false;
|
|
|
|
|
|
|
2025-11-29 15:15:22 +09:00
|
|
|
|
// 関数名ガードチェック
|
|
|
|
|
|
let func_name = self
|
|
|
|
|
|
.current_function
|
|
|
|
|
|
.as_ref()
|
|
|
|
|
|
.map(|f| f.signature.name.as_str())
|
|
|
|
|
|
.unwrap_or("");
|
|
|
|
|
|
let is_target = crate::mir::join_ir::lowering::is_joinir_if_toplevel_target(func_name);
|
|
|
|
|
|
|
2025-12-02 13:45:47 +09:00
|
|
|
|
// Phase 80: Core ON 時は代表関数で JoinIR を本線として試行
|
|
|
|
|
|
let core_mainline =
|
|
|
|
|
|
crate::mir::join_ir::lowering::should_try_joinir_mainline(func_name, false);
|
|
|
|
|
|
let strict_mode =
|
|
|
|
|
|
crate::mir::join_ir::lowering::should_panic_on_joinir_failure(func_name, false);
|
|
|
|
|
|
|
|
|
|
|
|
// Core ON + 本線対象の場合は環境変数に関わらず試行
|
2025-12-11 20:54:33 +09:00
|
|
|
|
let should_try_joinir =
|
|
|
|
|
|
core_mainline || (joinir_enabled && is_target && (joinir_toplevel || joinir_dryrun));
|
2025-12-02 13:45:47 +09:00
|
|
|
|
|
|
|
|
|
|
if should_try_joinir {
|
2025-11-29 15:08:23 +09:00
|
|
|
|
if let Some(ref func) = self.current_function {
|
|
|
|
|
|
let context =
|
|
|
|
|
|
crate::mir::join_ir::lowering::if_phi_context::IfPhiContext::pure_if();
|
|
|
|
|
|
|
|
|
|
|
|
match crate::mir::join_ir::lowering::try_lower_if_to_joinir(
|
|
|
|
|
|
func,
|
|
|
|
|
|
pre_branch_bb,
|
|
|
|
|
|
false,
|
|
|
|
|
|
Some(&context),
|
|
|
|
|
|
) {
|
|
|
|
|
|
Some(join_inst) => {
|
|
|
|
|
|
if joinir_dryrun || joinir_toplevel {
|
|
|
|
|
|
eprintln!(
|
2025-11-29 15:15:22 +09:00
|
|
|
|
"[Phase 61-4] ✅ Toplevel If lowered via JoinIR ({}): {:?}",
|
|
|
|
|
|
func_name, join_inst
|
2025-11-29 15:08:23 +09:00
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// PhiSpec 計算
|
|
|
|
|
|
let phi_spec = crate::mir::join_ir::lowering::if_phi_spec::compute_phi_spec_from_joinir(&context, &join_inst);
|
|
|
|
|
|
|
|
|
|
|
|
if joinir_dryrun {
|
|
|
|
|
|
eprintln!(
|
|
|
|
|
|
"[Phase 61-4] 🔍 dry-run: JoinIR PhiSpec header={}, exit={}",
|
|
|
|
|
|
phi_spec.header_count(),
|
|
|
|
|
|
phi_spec.exit_count()
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-11-29 15:32:40 +09:00
|
|
|
|
// Phase 61-4-F: 本番経路 - emit_toplevel_phis でPHI生成
|
2025-11-29 15:08:23 +09:00
|
|
|
|
if joinir_toplevel {
|
2025-11-29 15:32:40 +09:00
|
|
|
|
// IfShape 構築
|
|
|
|
|
|
let if_shape = IfShape {
|
|
|
|
|
|
cond_block: pre_branch_bb,
|
|
|
|
|
|
then_block,
|
|
|
|
|
|
else_block: Some(else_block),
|
|
|
|
|
|
merge_block,
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// PHI 生成
|
|
|
|
|
|
let phi_count = {
|
|
|
|
|
|
let mut ops = ToplevelOps(self);
|
2025-12-04 23:51:49 +09:00
|
|
|
|
crate::mir::if_in_loop_phi::IfInLoopPhiEmitter::emit_toplevel_phis(
|
2025-11-29 15:32:40 +09:00
|
|
|
|
&phi_spec,
|
|
|
|
|
|
&pre_if_var_map,
|
|
|
|
|
|
&then_var_map_end,
|
|
|
|
|
|
else_var_map_end_opt.as_ref(),
|
|
|
|
|
|
&mut ops,
|
|
|
|
|
|
&if_shape,
|
|
|
|
|
|
)?
|
|
|
|
|
|
};
|
|
|
|
|
|
|
2025-11-29 15:08:23 +09:00
|
|
|
|
if joinir_dryrun {
|
2025-11-29 15:32:40 +09:00
|
|
|
|
eprintln!(
|
|
|
|
|
|
"[Phase 61-4] ✅ Production path: {} PHIs generated via JoinIR",
|
|
|
|
|
|
phi_count
|
|
|
|
|
|
);
|
2025-11-29 15:08:23 +09:00
|
|
|
|
}
|
2025-11-29 15:32:40 +09:00
|
|
|
|
|
|
|
|
|
|
joinir_success = true;
|
2025-11-29 15:08:23 +09:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
None => {
|
|
|
|
|
|
if joinir_dryrun {
|
2025-11-30 09:38:28 +09:00
|
|
|
|
eprintln!(
|
|
|
|
|
|
"[Phase 61-4] ⏭️ JoinIR pattern not matched for {}, using fallback",
|
|
|
|
|
|
func_name
|
|
|
|
|
|
);
|
2025-11-29 15:08:23 +09:00
|
|
|
|
}
|
2025-12-02 13:45:47 +09:00
|
|
|
|
// Phase 80/81: Strict mode では本線対象関数の失敗でパニック
|
|
|
|
|
|
if strict_mode {
|
|
|
|
|
|
panic!(
|
|
|
|
|
|
"[joinir/if] strict mode: pattern not matched for {} (if_form.rs)",
|
|
|
|
|
|
func_name
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
2025-11-29 15:08:23 +09:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Phase 61-4: JoinIR成功時はスキップ、失敗時は既存経路
|
|
|
|
|
|
let result_val = if joinir_success {
|
|
|
|
|
|
// JoinIR 経路(未実装 - フォールバック)
|
|
|
|
|
|
crate::mir::builder::emission::constant::emit_void(self)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
self.normalize_if_else_phi(
|
|
|
|
|
|
then_block,
|
|
|
|
|
|
else_block,
|
|
|
|
|
|
if then_reaches_merge {
|
|
|
|
|
|
Some(then_exit_block)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
None
|
|
|
|
|
|
},
|
|
|
|
|
|
if else_reaches_merge {
|
|
|
|
|
|
Some(else_exit_block)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
None
|
|
|
|
|
|
},
|
|
|
|
|
|
then_value_raw,
|
|
|
|
|
|
else_value_raw,
|
|
|
|
|
|
&pre_if_var_map,
|
|
|
|
|
|
&then_ast_for_analysis,
|
|
|
|
|
|
&else_ast_for_analysis,
|
|
|
|
|
|
&then_var_map_end,
|
|
|
|
|
|
&else_var_map_end_opt,
|
|
|
|
|
|
pre_then_var_value,
|
|
|
|
|
|
)?
|
|
|
|
|
|
};
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
2025-09-21 08:53:00 +09:00
|
|
|
|
// Hint: join result variable(s)
|
|
|
|
|
|
// 1) Primary: if both branches assign to the same variable name, emit a hint for that name
|
2025-09-20 03:37:20 +09:00
|
|
|
|
if let (Some(tn), Some(en)) = (assigned_then_pre.as_deref(), assigned_else_pre.as_deref()) {
|
2025-11-21 06:25:17 +09:00
|
|
|
|
if tn == en {
|
|
|
|
|
|
self.hint_join_result(tn);
|
|
|
|
|
|
}
|
2025-09-21 08:53:00 +09:00
|
|
|
|
}
|
|
|
|
|
|
// 2) Secondary: if both branches assign multiple variables, hint全件(制限なし)
|
|
|
|
|
|
if let Some(ref else_map_end) = else_var_map_end_opt {
|
|
|
|
|
|
for name in then_var_map_end.keys() {
|
2025-11-21 06:25:17 +09:00
|
|
|
|
if Some(name.as_str()) == assigned_then_pre.as_deref() {
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
2025-09-21 08:53:00 +09:00
|
|
|
|
if else_map_end.contains_key(name) {
|
|
|
|
|
|
self.hint_join_result(name.as_str());
|
|
|
|
|
|
}
|
2025-09-20 03:37:20 +09:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-19 02:07:38 +09:00
|
|
|
|
// Merge other modified variables (skip the primary assignment if any)
|
2025-11-29 15:08:23 +09:00
|
|
|
|
if !joinir_success {
|
|
|
|
|
|
let skip_name = assigned_then_pre.as_deref();
|
|
|
|
|
|
self.merge_modified_vars(
|
|
|
|
|
|
then_block,
|
|
|
|
|
|
else_block,
|
|
|
|
|
|
if then_reaches_merge {
|
|
|
|
|
|
Some(then_exit_block)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
None
|
|
|
|
|
|
},
|
|
|
|
|
|
if else_reaches_merge {
|
|
|
|
|
|
Some(else_exit_block)
|
|
|
|
|
|
} else {
|
|
|
|
|
|
None
|
|
|
|
|
|
},
|
|
|
|
|
|
&pre_if_var_map,
|
|
|
|
|
|
&then_var_map_end,
|
|
|
|
|
|
&else_var_map_end_opt,
|
|
|
|
|
|
skip_name,
|
|
|
|
|
|
)?;
|
|
|
|
|
|
}
|
2025-09-19 02:07:38 +09:00
|
|
|
|
|
|
|
|
|
|
self.pop_if_merge();
|
2025-09-28 01:33:58 +09:00
|
|
|
|
// Pop merge debug region
|
|
|
|
|
|
self.debug_pop_region();
|
2025-09-19 02:07:38 +09:00
|
|
|
|
Ok(result_val)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|