A案実装: debug_verify_phi_inputs呼び出し前にCFG predecessorを更新
修正箇所(7箇所):
- src/mir/builder/phi.rs:50, 73, 132, 143
- src/mir/builder/ops.rs:273, 328, 351
根本原因:
- Branch/Jump命令でsuccessorは即座に更新
- predecessorはupdate_cfg()で遅延再構築
- PHI検証が先に実行されてpredecessor未更新でpanic
解決策:
- 各debug_verify_phi_inputs呼び出し前に
if let Some(func) = self.current_function.as_mut() {
func.update_cfg();
}
を挿入してCFGを同期
影響: if/else文、論理演算子(&&/||)のPHI生成が正常動作
577 lines
26 KiB
Plaintext
577 lines
26 KiB
Plaintext
using "lang/src/compiler/pipeline_v2/stage1_extract_flow.hako" as Stage1ExtractFlow
|
||
using "lang/src/compiler/pipeline_v2/emit_return_box.hako" as EmitReturnBox
|
||
using "lang/src/compiler/pipeline_v2/emit_binop_box.hako" as EmitBinopBox
|
||
using "lang/src/compiler/pipeline_v2/emit_compare_box.hako" as EmitCompareBox
|
||
using "lang/src/compiler/pipeline_v2/regex_flow.hako" as RegexFlow
|
||
using lang.compiler.builder.ssa.local as LocalSSA
|
||
using "lang/src/compiler/pipeline_v2/emit_call_box.hako" as EmitCallBox
|
||
using "lang/src/compiler/pipeline_v2/emit_method_box.hako" as EmitMethodBox
|
||
using "lang/src/compiler/pipeline_v2/emit_newbox_box.hako" as EmitNewBoxBox
|
||
using "lang/src/compiler/pipeline_v2/mir_call_box.hako" as MirCallBox
|
||
using "lang/src/shared/json/mir_v1_adapter.hako" as MirJsonV1Adapter
|
||
using "lang/src/compiler/pipeline_v2/compare_extract_box.hako" as CompareExtractBox
|
||
using "lang/src/compiler/pipeline_v2/normalizer_box.hako" as NormalizerBox
|
||
using "lang/src/compiler/pipeline_v2/map_helpers_box.hako" as MapHelpersBox
|
||
using "lang/src/compiler/pipeline_v2/call_extract_box.hako" as CallExtractBox
|
||
using "lang/src/compiler/pipeline_v2/method_extract_box.hako" as MethodExtractBox
|
||
using "lang/src/compiler/pipeline_v2/new_extract_box.hako" as NewExtractBox
|
||
using "lang/src/shared/common/string_helpers.hako" as StringHelpers
|
||
using "lang/src/compiler/pipeline_v2/using_resolver_box.hako" as UsingResolverBox
|
||
using "lang/src/compiler/pipeline_v2/namespace_box.hako" as NamespaceBox
|
||
using "lang/src/compiler/pipeline_v2/signature_verifier_box.hako" as SignatureVerifierBox
|
||
using "lang/src/compiler/pipeline_v2/stage1_json_scanner_box.hako" as Stage1JsonScannerBox
|
||
using "lang/src/compiler/pipeline_v2/stage1_name_args_normalizer_box.hako" as NameArgsNormBox
|
||
using "lang/src/compiler/pipeline_v2/alias_preflight_box.hako" as AliasPreflightBox
|
||
using "lang/src/compiler/pipeline_v2/stage1_args_parser_box.hako" as Stage1ArgsParserBox
|
||
using "lang/src/compiler/pipeline_v2/pipeline_helpers_box.hako" as PipelineHelpersBox
|
||
using "lang/src/shared/json/mir_v1_meta_inject_box.hako" as MirV1MetaInjectBox
|
||
|
||
flow PipelineV2 {
|
||
lower_stage1_to_mir(ast_json, prefer_cfg) {
|
||
// Backward-compatible entry (trace=0)
|
||
return PipelineV2.lower_stage1_to_mir_trace(ast_json, prefer_cfg, 0)
|
||
}
|
||
|
||
// Experimental: emit JSON v1 (MirCall) directly (no adapter). Shape-only focus.
|
||
lower_stage1_to_mir_v1(ast_json, prefer_cfg) {
|
||
if ast_json == null { return EmitReturnBox.emit_return_int2(0, 0) }
|
||
local r = UsingResolverBox.state_new()
|
||
// Call (prefer extract box)
|
||
{
|
||
local kq = RegexFlow.find_from(ast_json, "\"type\":\"Call\"", 0)
|
||
if kq >= 0 {
|
||
// Strict preflight (scanner): read name and enforce using alias resolution
|
||
{
|
||
local scan = Stage1JsonScannerBox.extract_name_args(ast_json, kq)
|
||
if scan != null {
|
||
if AliasPreflightBox.check_head(scan.get("name"), r) != 1 { return null }
|
||
}
|
||
}
|
||
|
||
local kc = CallExtractBox.extract_return_call_ints(ast_json)
|
||
if kc != null {
|
||
// Normalize to fix key/types at entry
|
||
local kn = NormalizerBox.normalize_call_ints(kc)
|
||
if kn == null { return null }
|
||
if SignatureVerifierBox.verify_call_name_arity(kn.get("name"), kn.get("args")) != 1 { return null }
|
||
local out_call = EmitCallBox.emit_call_int_args_v1(kn.get("name"), kn.get("args"))
|
||
if out_call == null { return null }
|
||
return out_call
|
||
}
|
||
// Tolerant scanner fallback → v1 emit (raw; no resolver context here)
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pair = Stage1JsonScannerBox.extract_name_args(ast_json, pos_body)
|
||
if pair != null {
|
||
local out_scan = EmitCallBox.emit_call_int_args_v1(pair.get("name"), pair.get("args_text"))
|
||
if out_scan == null { return null }
|
||
return out_scan
|
||
}
|
||
}
|
||
// Fallback legacy
|
||
local k = Stage1ExtractFlow.extract_return_call(ast_json)
|
||
if k != null {
|
||
local kn2 = NormalizerBox.normalize_call_ints(k)
|
||
if kn2 == null { return null }
|
||
if SignatureVerifierBox.verify_call_name_arity(kn2.get("name"), kn2.get("args")) != 1 { return null }
|
||
local out_legacy = EmitCallBox.emit_call_int_args_v1(kn2.get("name"), kn2.get("args"))
|
||
if out_legacy == null { return null }
|
||
return out_legacy
|
||
}
|
||
}
|
||
}
|
||
// Method (recv:0 placeholder; prefer extract box)
|
||
{
|
||
local mq = RegexFlow.find_from(ast_json, "\"type\":\"Method\"", 0)
|
||
if mq >= 0 {
|
||
local mb = MethodExtractBox.extract_return_method_ints(ast_json)
|
||
if mb != null {
|
||
local mn = NormalizerBox.normalize_method_ints(mb)
|
||
if mn == null { return null }
|
||
local out_method = EmitMethodBox.emit_method_int_args_v1(mn.get("method"), 0, mn.get("args"))
|
||
if out_method == null { return null }
|
||
return out_method
|
||
}
|
||
// Tolerant scanner fallback (method/args only)
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pair = Stage1JsonScannerBox.extract_label_args(ast_json, "method", pos_body)
|
||
if pair != null {
|
||
local out_method_scan = EmitMethodBox.emit_method_int_args_v1(pair.get("label"), 0, pair.get("args_text"))
|
||
if out_method_scan == null { return null }
|
||
return out_method_scan
|
||
}
|
||
}
|
||
local m = Stage1ExtractFlow.extract_return_method(ast_json)
|
||
if m != null {
|
||
local mn2 = NormalizerBox.normalize_method_ints(m)
|
||
if mn2 == null { return null }
|
||
local out_method_legacy = EmitMethodBox.emit_method_int_args_v1(mn2.get("method"), 0, mn2.get("args"))
|
||
if out_method_legacy == null { return null }
|
||
return out_method_legacy
|
||
}
|
||
}
|
||
}
|
||
// New (prefer extract box)
|
||
{
|
||
local nq = RegexFlow.find_from(ast_json, "\"type\":\"New\"", 0)
|
||
if nq >= 0 {
|
||
local nb = NewExtractBox.extract_return_new_ints(ast_json)
|
||
if nb != null {
|
||
local nn = NormalizerBox.normalize_new_ints(nb)
|
||
if nn == null { return null }
|
||
local out_new = EmitNewBoxBox.emit_newbox_int_args_v1(nn.get("class"), nn.get("args"))
|
||
if out_new == null { return null }
|
||
return out_new
|
||
}
|
||
// Tolerant scanner fallback (class/args only)
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pair = Stage1JsonScannerBox.extract_label_args(ast_json, "class", pos_body)
|
||
if pair != null {
|
||
local out_new_scan = EmitNewBoxBox.emit_newbox_int_args_v1(pair.get("label"), pair.get("args_text"))
|
||
if out_new_scan == null { return null }
|
||
return out_new_scan
|
||
}
|
||
}
|
||
local n = Stage1ExtractFlow.extract_return_new(ast_json)
|
||
if n != null {
|
||
local nn2 = NormalizerBox.normalize_new_ints(n)
|
||
if nn2 == null { return null }
|
||
local out_new_legacy = EmitNewBoxBox.emit_newbox_int_args_v1(nn2.get("class"), nn2.get("args"))
|
||
if out_new_legacy == null { return null }
|
||
return out_new_legacy
|
||
}
|
||
}
|
||
}
|
||
// Fallback to v0 path for remaining cases
|
||
return PipelineV2.lower_stage1_to_mir(ast_json, prefer_cfg)
|
||
}
|
||
|
||
// Experimental (with metadata): emit JSON v1 and inject metadata.extern_c.
|
||
// externs_json: JSON array text like
|
||
// [{"func":"Name/Arity","symbol":"c_symbol"}, ...]
|
||
lower_stage1_to_mir_v1_with_meta(ast_json, prefer_cfg, externs_json) {
|
||
local j1 = PipelineV2.lower_stage1_to_mir_v1(ast_json, prefer_cfg)
|
||
return MirV1MetaInjectBox.inject_meta_externs(j1, externs_json)
|
||
}
|
||
|
||
// Experimental helper: emit v1 then downgrade to v0 for Mini‑VM exec
|
||
lower_stage1_to_mir_v1_compat(ast_json, prefer_cfg) {
|
||
local j1 = PipelineV2.lower_stage1_to_mir_v1(ast_json, prefer_cfg)
|
||
return MirJsonV1Adapter.to_v0(j1)
|
||
}
|
||
|
||
lower_stage1_to_mir_trace(ast_json, prefer_cfg, trace) {
|
||
local prefer = PipelineHelpersBox.to_i64(prefer_cfg)
|
||
if ast_json == null { return EmitReturnBox.emit_return_int2(0, trace) }
|
||
if trace == 1 { print("[pipe] prefer_cfg=" + prefer_cfg + " prefer=" + prefer) }
|
||
local r = UsingResolverBox.state_new()
|
||
// Early fast-path: any Compare in program (Int-only)
|
||
{
|
||
local cq0 = RegexFlow.find_from(ast_json, "\"type\":\"Compare\"", 0)
|
||
if cq0 >= 0 {
|
||
local opk_pos0 = RegexFlow.find_from(ast_json, "\"op\":\"", cq0)
|
||
local cmp0 = ""
|
||
if opk_pos0 >= 0 {
|
||
local opk_end0 = RegexFlow.find_from(ast_json, "\"", opk_pos0 + 6)
|
||
if opk_end0 >= 0 { cmp0 = ast_json.substring(opk_pos0 + 6, opk_end0) }
|
||
}
|
||
local lhsp0 = RegexFlow.find_from(ast_json, "\"lhs\"", cq0)
|
||
local rhsp0 = RegexFlow.find_from(ast_json, "\"rhs\"", cq0)
|
||
if lhsp0 >= 0 && rhsp0 >= 0 {
|
||
local lv0 = 0
|
||
local rv0 = 0
|
||
local vpos0 = RegexFlow.find_from(ast_json, "\"value\":", lhsp0)
|
||
if vpos0 >= 0 {
|
||
local ds0 = RegexFlow.digits_from(ast_json, vpos0 + 8)
|
||
if ds0 != "" { lv0 = RegexFlow.to_int(ds0) }
|
||
}
|
||
local vpos02 = RegexFlow.find_from(ast_json, "\"value\":", rhsp0)
|
||
if vpos02 >= 0 {
|
||
local ds02 = RegexFlow.digits_from(ast_json, vpos02 + 8)
|
||
if ds02 != "" { rv0 = RegexFlow.to_int(ds02) }
|
||
}
|
||
if cmp0 == null || cmp0 == "" { cmp0 = "Gt" }
|
||
if prefer >= 1 {
|
||
local mat0 = 0
|
||
if prefer >= 2 { mat0 = 1 }
|
||
if trace == 1 { print("[pipe] any-fast cfg cmp=" + cmp0 + " lhs=" + (""+lv0) + " rhs=" + (""+rv0) + " mat=" + (""+mat0)) }
|
||
local jfast0 = EmitCompareBox.emit_compare_cfg3(lv0, rv0, cmp0, mat0, trace)
|
||
return LocalSSA.ensure_cond(jfast0)
|
||
} else {
|
||
if trace == 1 { print("[pipe] any-fast ret cmp=" + cmp0 + " lhs=" + (""+lv0) + " rhs=" + (""+rv0)) }
|
||
return EmitCompareBox.emit_compare_ret(lv0, rv0, cmp0, trace)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// If(...) pattern (prefer CFG form)
|
||
{
|
||
local iq = RegexFlow.find_from(ast_json, "\"type\":\"If\"", 0)
|
||
if iq >= 0 {
|
||
// Fast-path: directly parse Compare within If for primitives (Int-only)
|
||
{
|
||
local cv = PipelineHelpersBox.parse_compare_values(ast_json, iq)
|
||
if cv != null {
|
||
local mat = 1
|
||
local lfast = cv.get(0)
|
||
local rfast = cv.get(1)
|
||
local cmpfast = cv.get(2)
|
||
if cmpfast == null || cmpfast == "" { cmpfast = "Gt" }
|
||
if trace == 1 { print("[pipe] if-fast cmp=" + cmpfast + " lhs=" + (""+lfast) + " rhs=" + (""+rfast) + " mat=" + (""+mat)) }
|
||
local jfast = EmitCompareBox.emit_compare_cfg3(lfast, rfast, cmpfast, mat, trace)
|
||
return LocalSSA.ensure_cond(jfast)
|
||
}
|
||
}
|
||
// Prefer robust extractor box
|
||
local icb = CompareExtractBox.extract_if_compare_ints(ast_json)
|
||
if icb != null {
|
||
local mat = 0
|
||
if prefer >= 2 { mat = 1 }
|
||
local l = icb.get(0)
|
||
local r = icb.get(1)
|
||
local c = icb.get(2)
|
||
local j0 = EmitCompareBox.emit_compare_cfg3(l, r, c, mat, trace)
|
||
return LocalSSA.ensure_cond(j0)
|
||
}
|
||
// Fallback legacy extractor
|
||
local ic = Stage1ExtractFlow.extract_if_compare(ast_json)
|
||
if ic != null {
|
||
local mat = 0
|
||
if prefer >= 2 { mat = 1 }
|
||
local l2 = ic.get("lhs") + 0
|
||
local r2 = ic.get("rhs") + 0
|
||
local c2 = "" + ic.get("cmp")
|
||
if c2 == null || c2 == "" { c2 = "Gt" }
|
||
local j0 = EmitCompareBox.emit_compare_cfg3(l2, r2, c2, mat, trace)
|
||
return LocalSSA.ensure_cond(j0)
|
||
}
|
||
}
|
||
}
|
||
// Next Call (Return(Call ...)) — prefer extract box
|
||
{
|
||
local kq = RegexFlow.find_from(ast_json, "\"type\":\"Call\"", 0)
|
||
if kq >= 0 {
|
||
print("[flow] Call pattern: kq=" + kq)
|
||
// Strict preflight via tolerant scanner: read raw name and enforce using alias resolution
|
||
{
|
||
local scan0 = Stage1JsonScannerBox.extract_name_args(ast_json, kq)
|
||
if scan0 != null {
|
||
print("[flow] Call scan0 name=" + scan0.get("name"))
|
||
if AliasPreflightBox.check_head(scan0.get("name"), r) != 1 {
|
||
print("[flow] Call AliasPreflightBox failed, returning null")
|
||
return null
|
||
}
|
||
}
|
||
}
|
||
|
||
local kc = CallExtractBox.extract_return_call_ints(ast_json)
|
||
if kc != null {
|
||
print("[flow] Call extract_return_call_ints succeeded")
|
||
local kn = NormalizerBox.normalize_call_ints(kc)
|
||
if kn == null { print("[flow] Call normalize_call_ints returned null") return null }
|
||
if SignatureVerifierBox.verify_call_name_arity(kn.get("name"), kn.get("args")) != 1 { print("[flow] Call verify_call_name_arity failed") return null }
|
||
local j4 = EmitCallBox.emit_call_int_args(kn.get("name"), kn.get("args"))
|
||
if j4 == null { print("[flow] Call emit_call_int_args returned null") return null }
|
||
print("[flow] Call path 1 succeeded, returning JSON")
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j4))
|
||
} else {
|
||
print("[flow] Call extract_return_call_ints returned null, trying scanner fallback")
|
||
}
|
||
// Fallback: scanner → normalizer → emit
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pair = Stage1JsonScannerBox.extract_name_args(ast_json, pos_body)
|
||
if pair != null {
|
||
local norm = NameArgsNormBox.normalize_call(pair, r)
|
||
if norm != null {
|
||
local j4b = EmitCallBox.emit_call_int_args(norm.get("name"), norm.get("args_text"))
|
||
if j4b == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j4b))
|
||
}
|
||
}
|
||
}
|
||
|
||
local k = Stage1ExtractFlow.extract_return_call(ast_json)
|
||
if k != null {
|
||
local kn2 = NormalizerBox.normalize_call_ints(k)
|
||
if kn2 == null { return null }
|
||
if SignatureVerifierBox.verify_call_name_arity(kn2.get("name"), kn2.get("args")) != 1 { return null }
|
||
local j4 = EmitCallBox.emit_call_int_args(kn2.get("name"), kn2.get("args"))
|
||
if j4 == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j4))
|
||
}
|
||
// (scanner fallback handled above)
|
||
|
||
}
|
||
}
|
||
// Next Method (Return(Method ...)) — prefer extract box; recv is 0 (placeholder)
|
||
{
|
||
local mq = RegexFlow.find_from(ast_json, "\"type\":\"Method\"", 0)
|
||
if mq >= 0 {
|
||
local mb = MethodExtractBox.extract_return_method_ints(ast_json)
|
||
if mb != null {
|
||
local mn = NormalizerBox.normalize_method_ints(mb)
|
||
if mn == null { return null }
|
||
// Compile-time arity check (best-effort by method name)
|
||
if SignatureVerifierBox.verify_from_args(mn.get("method"), mn.get("args")) != 1 { return null }
|
||
local j5 = EmitMethodBox.emit_method_int_args(mn.get("method"), 0, mn.get("args"))
|
||
if j5 == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j5))
|
||
}
|
||
// Lightweight Stage1 scanner as tolerant fallback (method/args only)
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pair = Stage1JsonScannerBox.extract_label_args(ast_json, "method", pos_body)
|
||
if pair != null {
|
||
local raw2 = pair.get("label")
|
||
local args2 = pair.get("args_text")
|
||
local fq2 = NamespaceBox.normalize_global_name(raw2, r)
|
||
if fq2 != null {
|
||
if SignatureVerifierBox.verify_from_args(fq2, args2) == 1 {
|
||
local j5b = EmitMethodBox.emit_method_int_args(fq2, 0, args2)
|
||
if j5b == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j5b))
|
||
}
|
||
}
|
||
}
|
||
}
|
||
local m = Stage1ExtractFlow.extract_return_method(ast_json)
|
||
if m != null {
|
||
local mn2 = NormalizerBox.normalize_method_ints(m)
|
||
if mn2 == null { return null }
|
||
if SignatureVerifierBox.verify_from_args(mn2.get("method"), mn2.get("args")) != 1 { return null }
|
||
local j5 = EmitMethodBox.emit_method_int_args(mn2.get("method"), 0, mn2.get("args"))
|
||
if j5 == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j5))
|
||
}
|
||
}
|
||
}
|
||
// Next New (Return(New ...)) — prefer extract box
|
||
{
|
||
local nq = RegexFlow.find_from(ast_json, "\"type\":\"New\"", 0)
|
||
if nq >= 0 {
|
||
local nb = NewExtractBox.extract_return_new_ints(ast_json)
|
||
if nb != null {
|
||
local nn = NormalizerBox.normalize_new_ints(nb)
|
||
if nn == null { return null }
|
||
local j6 = EmitNewBoxBox.emit_newbox_int_args(nn.get("class"), nn.get("args"))
|
||
if j6 == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j6))
|
||
}
|
||
local n = Stage1ExtractFlow.extract_return_new(ast_json)
|
||
if n != null {
|
||
local nn2 = NormalizerBox.normalize_new_ints(n)
|
||
if nn2 == null { return null }
|
||
local j6 = EmitNewBoxBox.emit_newbox_int_args(nn2.get("class"), nn2.get("args"))
|
||
if j6 == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j6))
|
||
}
|
||
// Tolerant scanner for New: class/args only
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pairN = Stage1JsonScannerBox.extract_label_args(ast_json, "class", pos_body)
|
||
if pairN != null {
|
||
local rawC = pairN.get("label")
|
||
local argsN = pairN.get("args_text")
|
||
// No namespace resolution here (non-using path): emit directly
|
||
local j6b = EmitNewBoxBox.emit_newbox_int_args(rawC, argsN)
|
||
if j6b == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j6b))
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// Try Compare first (avoid mis-detection by 'op' field inside Compare)
|
||
// Guard by raw scan to avoid calling methods on null-like values
|
||
{
|
||
local cq = RegexFlow.find_from(ast_json, "\"type\":\"Compare\"", 0)
|
||
if cq >= 0 {
|
||
// Prefer robust extractor box (returns ArrayBox [lv, rv, cmp])
|
||
local ce = CompareExtractBox.extract_return_compare_ints(ast_json)
|
||
if ce != null {
|
||
local final_lhs = ce.get(0) // ArrayBox: lhs at index 0
|
||
local final_rhs = ce.get(1) // ArrayBox: rhs at index 1
|
||
local cmp = ce.get(2) // ArrayBox: cmp at index 2
|
||
if trace == 1 { print("[trace] compare via box lhs=" + final_lhs + " rhs=" + final_rhs + " cmp=" + cmp) }
|
||
if prefer_cfg >= 1 {
|
||
local mat = 0
|
||
if prefer >= 2 { mat = 1 }
|
||
// Direct emit (no MapBox/Normalizer needed - values already normalized)
|
||
local j1 = EmitCompareBox.emit_compare_cfg3(final_lhs, final_rhs, cmp, mat, trace)
|
||
return LocalSSA.ensure_cond(j1)
|
||
} else {
|
||
// Direct emit (no MapBox/Normalizer needed - values already normalized)
|
||
local j0 = EmitCompareBox.emit_compare_ret(final_lhs, final_rhs, cmp, trace)
|
||
return LocalSSA.ensure_cond(j0)
|
||
}
|
||
}
|
||
// Fallback to legacy extractor (should rarely be used)
|
||
local c = Stage1ExtractFlow.extract_return_compare(ast_json)
|
||
if c != null {
|
||
local lhs = c.get("lhs")
|
||
local rhs = c.get("rhs")
|
||
local cmp2 = c.get("cmp")
|
||
if trace == 1 { print("[trace] compare via legacy lhs=" + lhs + " rhs=" + rhs + " cmp=" + cmp2) }
|
||
if prefer_cfg >= 1 {
|
||
local mat = 0
|
||
if prefer >= 2 { mat = 1 }
|
||
local raw3 = {cmp: cmp2, lhs: lhs, rhs: rhs}
|
||
local nn3 = NormalizerBox.normalize_cmp(raw3)
|
||
if nn3 == null { return null }
|
||
local j1 = EmitCompareBox.emit_compare_cfg3(nn3.get("lhs"), nn3.get("rhs"), nn3.get("cmp"), mat, trace)
|
||
return LocalSSA.ensure_cond(j1)
|
||
} else {
|
||
local raw4 = {cmp: cmp2, lhs: lhs, rhs: rhs}
|
||
local nn4 = NormalizerBox.normalize_cmp(raw4)
|
||
if nn4 == null { return null }
|
||
local j0 = EmitCompareBox.emit_compare_ret(nn4.get("lhs"), nn4.get("rhs"), nn4.get("cmp"), trace)
|
||
return LocalSSA.ensure_cond(j0)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// Then BinOp (guard by raw scan)
|
||
{
|
||
local bq = RegexFlow.find_from(ast_json, "\"type\":\"BinOp\"", 0)
|
||
if bq >= 0 {
|
||
local b = Stage1ExtractFlow.extract_return_binop(ast_json)
|
||
local lhs = b.get("lhs")
|
||
local rhs = b.get("rhs")
|
||
local kind = b.get("kind")
|
||
local j2 = EmitBinopBox.emit_binop2(lhs, rhs, kind, trace)
|
||
return LocalSSA.ensure_cond(j2)
|
||
}
|
||
}
|
||
// Fallback Return(Int)
|
||
local v = Stage1ExtractFlow.extract_return_int(ast_json)
|
||
local j3 = EmitReturnBox.emit_return_int2(v, trace)
|
||
return LocalSSA.ensure_cond(j3)
|
||
}
|
||
// Overload: resolve names via UsingResolverBox before emit
|
||
lower_stage1_to_mir_with_usings(ast_json, prefer_cfg, usings_json, modules_json) {
|
||
print("[DEBUG] === lower_stage1_to_mir_with_usings ENTRY ===")
|
||
print("[DEBUG] ast_json length=" + ast_json.length())
|
||
print("[DEBUG] usings_json=" + usings_json)
|
||
print("[DEBUG] modules_json=" + modules_json)
|
||
if ast_json == null { return PipelineV2.lower_stage1_to_mir(ast_json, prefer_cfg) }
|
||
// Build resolver context
|
||
local r = UsingResolverBox.state_new()
|
||
if usings_json != null { UsingResolverBox.load_usings_json(r, usings_json) }
|
||
if modules_json != null { UsingResolverBox.load_modules_json(r, modules_json) }
|
||
// Upgrade alias→namespace mapping now that modules are loaded (Module‑First)
|
||
UsingResolverBox.upgrade_aliases(r)
|
||
print("[DEBUG] upgrade_aliases complete")
|
||
|
||
// Prefer Call/Method/New branches with name normalization; otherwise fallback to default
|
||
{
|
||
local kq = RegexFlow.find_from(ast_json, "\"type\":\"Call\"", 0)
|
||
if kq >= 0 {
|
||
// Strict preflight via tolerant scanner: read raw name and enforce using alias resolution
|
||
{
|
||
local scan0 = Stage1JsonScannerBox.extract_name_args(ast_json, kq)
|
||
if scan0 != null {
|
||
local raw_head = scan0.get("name")
|
||
print("[DEBUG] preflight raw_head=" + raw_head)
|
||
local dot = RegexFlow.find_from(raw_head, ".", 0)
|
||
if dot >= 0 {
|
||
local head = raw_head.substring(0, dot)
|
||
local resolved = UsingResolverBox.resolve_namespace_alias(r, head)
|
||
print("[DEBUG] preflight head=" + head + " resolved=" + resolved)
|
||
if resolved == null {
|
||
print("[ERROR] Unresolved using alias: " + head)
|
||
return null
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// Prefer lightweight Stage1 scanner first(plugins OFFでも安全)
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local pair = Stage1JsonScannerBox.extract_name_args(ast_json, pos_body)
|
||
if pair != null {
|
||
local raw2 = pair.get("name")
|
||
local args2 = pair.get("args_text")
|
||
local fq2 = NamespaceBox.normalize_global_name(raw2, r)
|
||
if fq2 != null {
|
||
local j4b = EmitCallBox.emit_call_int_args(fq2, args2)
|
||
if j4b == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j4b))
|
||
}
|
||
}
|
||
}
|
||
// Structured extract(配列/Map依存。plugins ON時の静的パス)
|
||
local kc = CallExtractBox.extract_return_call_ints(ast_json)
|
||
if kc != null {
|
||
local kn = NormalizerBox.normalize_call_ints(kc)
|
||
if kn == null {
|
||
print("[DEBUG] normalize_call_ints returned null")
|
||
return null
|
||
}
|
||
local raw_name = kn.get("name")
|
||
print("[DEBUG] raw_name before normalize=" + raw_name)
|
||
if SignatureVerifierBox.verify_call_name_arity(raw_name, kn.get("args")) != 1 {
|
||
print("[DEBUG] verify_call_name_arity failed for " + raw_name)
|
||
return null
|
||
}
|
||
local fq_name = NamespaceBox.normalize_global_name(raw_name, r)
|
||
print("[DEBUG] fq_name after normalize=" + fq_name)
|
||
if fq_name == null {
|
||
print("[DEBUG] normalize_global_name returned null for raw_name=" + raw_name)
|
||
return null
|
||
}
|
||
local j4 = EmitCallBox.emit_call_int_args(fq_name, kn.get("args"))
|
||
if j4 == null {
|
||
print("[DEBUG] emit_call_int_args returned null")
|
||
return null
|
||
}
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j4))
|
||
}
|
||
|
||
}
|
||
}
|
||
{
|
||
local nq = RegexFlow.find_from(ast_json, "\"type\":\"New\"", 0)
|
||
if nq >= 0 {
|
||
local nb = NewExtractBox.extract_return_new_ints(ast_json)
|
||
if nb != null {
|
||
local nn = NormalizerBox.normalize_new_ints(nb)
|
||
if nn == null { return null }
|
||
local raw_c = nn.get("class")
|
||
local fq_c = NamespaceBox.normalize_class_name(raw_c, r)
|
||
if fq_c == null { return null }
|
||
local j6 = EmitNewBoxBox.emit_newbox_int_args(fq_c, nn.get("args"))
|
||
if j6 == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j6))
|
||
}
|
||
// Scanner fallback: class + args → normalizer → emit
|
||
{
|
||
local pos_body = Stage1JsonScannerBox.find_body_start(ast_json)
|
||
if pos_body < 0 { pos_body = 0 }
|
||
local nscan = Stage1JsonScannerBox.extract_label_args(ast_json, "class", pos_body)
|
||
if nscan != null {
|
||
local normN = NameArgsNormBox.normalize_class(nscan, r)
|
||
if normN != null {
|
||
local j6b = EmitNewBoxBox.emit_newbox_int_args(normN.get("class"), normN.get("args_text"))
|
||
if j6b == null { return null }
|
||
return LocalSSA.ensure_calls(LocalSSA.ensure_cond(j6b))
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
// Fallback to default lowering (includes Compare/Method and other forms)
|
||
return PipelineV2.lower_stage1_to_mir(ast_json, prefer_cfg)
|
||
}
|
||
|
||
}
|