Files
hakorune/src/backend/mir_interpreter/handlers/boxes.rs

498 lines
23 KiB
Rust
Raw Normal View History

use super::*;
use crate::box_trait::NyashBox;
impl MirInterpreter {
pub(super) fn handle_new_box(
&mut self,
dst: ValueId,
box_type: &str,
args: &[ValueId],
) -> Result<(), VMError> {
// Provider Lock guard (受け口・既定は挙動不変)
if let Err(e) = crate::runtime::provider_lock::guard_before_new_box(box_type) {
return Err(VMError::InvalidInstruction(e));
}
let mut converted: Vec<Box<dyn NyashBox>> = Vec::with_capacity(args.len());
for vid in args {
converted.push(self.reg_load(*vid)?.to_nyash_box());
}
let reg = crate::runtime::unified_registry::get_global_unified_registry();
let created = reg
.lock()
.unwrap()
.create_box(box_type, &converted)
.map_err(|e| {
VMError::InvalidInstruction(format!("NewBox {} failed: {}", box_type, e))
})?;
// Store created instance first so 'me' can be passed to birth
let created_vm = VMValue::from_nyash_box(created);
self.regs.insert(dst, created_vm.clone());
// Trace: new box event (dev-only)
if Self::box_trace_enabled() {
self.box_trace_emit_new(box_type, args.len());
}
// Note: birth の自動呼び出しは削除。
// 正しい設計は Builder が NewBox 後に明示的に birth 呼び出しを生成すること。
Ok(())
}
pub(super) fn handle_plugin_invoke(
&mut self,
dst: Option<ValueId>,
box_val: ValueId,
method: &str,
args: &[ValueId],
) -> Result<(), VMError> {
let recv = self.reg_load(box_val)?;
let recv_box: Box<dyn NyashBox> = match recv.clone() {
VMValue::BoxRef(b) => b.share_box(),
other => other.to_nyash_box(),
};
if let Some(p) = recv_box
.as_any()
.downcast_ref::<crate::runtime::plugin_loader_v2::PluginBoxV2>()
{
let host = crate::runtime::plugin_loader_unified::get_global_plugin_host();
let host = host.read().unwrap();
let mut argv: Vec<Box<dyn NyashBox>> = Vec::with_capacity(args.len());
for a in args {
argv.push(self.reg_load(*a)?.to_nyash_box());
}
match host.invoke_instance_method(&p.box_type, method, p.inner.instance_id, &argv) {
Ok(Some(ret)) => {
if let Some(d) = dst {
self.regs.insert(d, VMValue::from_nyash_box(ret));
}
}
Ok(None) => {
if let Some(d) = dst {
self.regs.insert(d, VMValue::Void);
}
}
Err(e) => {
return Err(VMError::InvalidInstruction(format!(
"PluginInvoke {}.{} failed: {:?}",
p.box_type, method, e
)))
}
}
Ok(())
} else if method == "toString" {
if let Some(d) = dst {
self.regs
.insert(d, VMValue::String(recv_box.to_string_box().value));
}
Ok(())
} else {
Err(VMError::InvalidInstruction(format!(
"PluginInvoke unsupported on {} for method {}",
recv_box.type_name(),
method
)))
}
}
pub(super) fn handle_box_call(
&mut self,
dst: Option<ValueId>,
box_val: ValueId,
method: &str,
args: &[ValueId],
) -> Result<(), VMError> {
// Dev-safe: stringify(Void) → "null" (最小安全弁)
if method == "stringify" {
if let VMValue::Void = self.reg_load(box_val)? {
if let Some(d) = dst { self.regs.insert(d, VMValue::String("null".to_string())); }
return Ok(());
}
if let VMValue::BoxRef(b) = self.reg_load(box_val)? {
if b.as_any().downcast_ref::<crate::box_trait::VoidBox>().is_some() {
if let Some(d) = dst { self.regs.insert(d, VMValue::String("null".to_string())); }
return Ok(());
}
}
}
// Trace: method call (class inferred from receiver)
if Self::box_trace_enabled() {
let cls = match self.reg_load(box_val).unwrap_or(VMValue::Void) {
VMValue::BoxRef(b) => {
if let Some(inst) = b.as_any().downcast_ref::<crate::instance_v2::InstanceBox>() {
inst.class_name.clone()
} else {
b.type_name().to_string()
}
}
VMValue::String(_) => "StringBox".to_string(),
VMValue::Integer(_) => "IntegerBox".to_string(),
VMValue::Float(_) => "FloatBox".to_string(),
VMValue::Bool(_) => "BoolBox".to_string(),
VMValue::Void => "<Void>".to_string(),
VMValue::Future(_) => "<Future>".to_string(),
};
self.box_trace_emit_call(&cls, method, args.len());
}
// Debug: trace length dispatch receiver type before any handler resolution
if method == "length" && std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
let recv = self.reg_load(box_val).unwrap_or(VMValue::Void);
let type_name = match recv.clone() {
VMValue::BoxRef(b) => b.type_name().to_string(),
VMValue::Integer(_) => "Integer".to_string(),
VMValue::Float(_) => "Float".to_string(),
VMValue::Bool(_) => "Bool".to_string(),
VMValue::String(_) => "String".to_string(),
VMValue::Void => "Void".to_string(),
VMValue::Future(_) => "Future".to_string(),
};
eprintln!("[vm-trace] length dispatch recv_type={}", type_name);
}
// Graceful void guard for common short-circuit patterns in user code
// e.g., `A or not last.is_eof()` should not crash when last is absent.
match self.reg_load(box_val)? {
VMValue::Void => {
match method {
"is_eof" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Bool(false)); } return Ok(()); }
"length" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(0)); } return Ok(()); }
"substring" => { if let Some(d) = dst { self.regs.insert(d, VMValue::String(String::new())); } return Ok(()); }
"push" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Void); } return Ok(()); }
"get_position" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(0)); } return Ok(()); }
"get_line" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(1)); } return Ok(()); }
"get_column" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(1)); } return Ok(()); }
_ => {}
}
}
VMValue::BoxRef(ref b) => {
if b.as_any().downcast_ref::<crate::box_trait::VoidBox>().is_some() {
match method {
"is_eof" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Bool(false)); } return Ok(()); }
"length" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(0)); } return Ok(()); }
"substring" => { if let Some(d) = dst { self.regs.insert(d, VMValue::String(String::new())); } return Ok(()); }
"push" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Void); } return Ok(()); }
"get_position" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(0)); } return Ok(()); }
"get_line" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(1)); } return Ok(()); }
"get_column" => { if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(1)); } return Ok(()); }
_ => {}
}
}
}
_ => {}
}
if super::boxes_object_fields::try_handle_object_fields(self, dst, box_val, method, args)? {
if method == "length" && std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
eprintln!("[vm-trace] length dispatch handler=object_fields");
}
return Ok(());
}
// Policy gate: user InstanceBox BoxCall runtime fallback
// - Prod: disallowed (builder must have rewritten obj.m(...) to a
// function call). Error here indicates a builder/using materialize
// miss.
// - Dev/CI: allowed with WARN to aid diagnosis.
let mut user_instance_class: Option<String> = None;
if let VMValue::BoxRef(ref b) = self.reg_load(box_val)? {
if let Some(inst) = b.as_any().downcast_ref::<crate::instance_v2::InstanceBox>() {
user_instance_class = Some(inst.class_name.clone());
}
}
if user_instance_class.is_some() && !crate::config::env::vm_allow_user_instance_boxcall() {
let cls = user_instance_class.unwrap();
return Err(VMError::InvalidInstruction(format!(
"User Instance BoxCall disallowed in prod: {}.{} (enable builder rewrite)",
cls, method
)));
}
if user_instance_class.is_some() && crate::config::env::vm_allow_user_instance_boxcall() {
if crate::config::env::cli_verbose() {
eprintln!(
"[warn] dev fallback: user instance BoxCall {}.{} routed via VM instance-dispatch",
user_instance_class.as_ref().unwrap(),
method
);
}
}
if super::boxes_instance::try_handle_instance_box(self, dst, box_val, method, args)? {
if method == "length" && std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
eprintln!("[vm-trace] length dispatch handler=instance_box");
}
return Ok(());
}
if super::boxes_string::try_handle_string_box(self, dst, box_val, method, args)? {
if method == "length" && std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
eprintln!("[vm-trace] length dispatch handler=string_box");
}
return Ok(());
}
if super::boxes_array::try_handle_array_box(self, dst, box_val, method, args)? {
if method == "length" && std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
eprintln!("[vm-trace] length dispatch handler=array_box");
}
return Ok(());
}
if super::boxes_map::try_handle_map_box(self, dst, box_val, method, args)? {
if method == "length" && std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
eprintln!("[vm-trace] length dispatch handler=map_box");
}
return Ok(());
}
// Narrow safety valve: if 'length' wasn't handled by any box-specific path,
// treat it as 0 (avoids Lt on Void in common loops). This is a dev-time
// robustness measure; precise behavior should be provided by concrete boxes.
if method == "length" {
if std::env::var("NYASH_VM_TRACE").ok().as_deref() == Some("1") {
eprintln!("[vm-trace] length dispatch handler=fallback(length=0)");
}
if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(0)); }
return Ok(());
}
// Fallback: unique-tail dynamic resolution for user-defined methods
// Narrowing: restrict to receiver's class when available to avoid
// accidentally binding methods from unrelated boxes that happen to
// share the same method name/arity (e.g., JsonScanner.is_eof vs JsonToken.is_eof).
if let Some(func) = {
let tail = format!(".{}{}", method, format!("/{}", args.len()));
let mut cands: Vec<String> = self
.functions
.keys()
.filter(|k| k.ends_with(&tail))
.cloned()
.collect();
// Determine receiver class name when possible
let recv_cls: Option<String> = match self.reg_load(box_val).ok() {
Some(VMValue::BoxRef(b)) => {
if let Some(inst) = b.as_any().downcast_ref::<crate::instance_v2::InstanceBox>() {
Some(inst.class_name.clone())
} else { None }
}
_ => None,
};
if let Some(ref want) = recv_cls {
let prefix = format!("{}.", want);
cands.retain(|k| k.starts_with(&prefix));
}
if cands.len() == 1 { self.functions.get(&cands[0]).cloned() } else { None }
} {
// Build argv: pass receiver as first arg (me)
let recv_vm = self.reg_load(box_val)?;
let mut argv: Vec<VMValue> = Vec::with_capacity(1 + args.len());
argv.push(recv_vm);
for a in args { argv.push(self.reg_load(*a)?); }
let ret = self.exec_function_inner(&func, Some(&argv))?;
if let Some(d) = dst { self.regs.insert(d, ret); }
return Ok(());
}
self.invoke_plugin_box(dst, box_val, method, args)
}
// moved: try_handle_map_box → handlers/boxes_map.rs
fn try_handle_map_box(
&mut self,
dst: Option<ValueId>,
box_val: ValueId,
method: &str,
args: &[ValueId],
) -> Result<bool, VMError> {
super::boxes_map::try_handle_map_box(self, dst, box_val, method, args)
}
// moved: try_handle_string_box → handlers/boxes_string.rs
fn try_handle_string_box(
&mut self,
dst: Option<ValueId>,
box_val: ValueId,
method: &str,
args: &[ValueId],
) -> Result<bool, VMError> {
super::boxes_string::try_handle_string_box(self, dst, box_val, method, args)
}
// moved: try_handle_array_box → handlers/boxes_array.rs
fn try_handle_array_box(
&mut self,
dst: Option<ValueId>,
box_val: ValueId,
method: &str,
args: &[ValueId],
) -> Result<bool, VMError> {
super::boxes_array::try_handle_array_box(self, dst, box_val, method, args)
}
fn invoke_plugin_box(
&mut self,
dst: Option<ValueId>,
box_val: ValueId,
method: &str,
args: &[ValueId],
) -> Result<(), VMError> {
let recv = self.reg_load(box_val)?;
let recv_box: Box<dyn NyashBox> = match recv.clone() {
VMValue::BoxRef(b) => b.share_box(),
other => other.to_nyash_box(),
};
if let Some(p) = recv_box
.as_any()
.downcast_ref::<crate::runtime::plugin_loader_v2::PluginBoxV2>()
{
if p.box_type == "ConsoleBox" && method == "readLine" {
use std::io::{self, Read};
let mut s = String::new();
let mut stdin = io::stdin();
let mut buf = [0u8; 1];
while let Ok(n) = stdin.read(&mut buf) {
if n == 0 {
break;
}
let ch = buf[0] as char;
if ch == '\n' {
break;
}
s.push(ch);
if s.len() > 1_000_000 {
break;
}
}
if let Some(d) = dst {
self.regs.insert(d, VMValue::String(s));
}
return Ok(());
}
let host = crate::runtime::plugin_loader_unified::get_global_plugin_host();
let host = host.read().unwrap();
let mut argv: Vec<Box<dyn NyashBox>> = Vec::with_capacity(args.len());
for a in args {
argv.push(self.reg_load(*a)?.to_nyash_box());
}
match host.invoke_instance_method(&p.box_type, method, p.inner.instance_id, &argv) {
Ok(Some(ret)) => {
if let Some(d) = dst {
self.regs.insert(d, VMValue::from_nyash_box(ret));
}
Ok(())
}
Ok(None) => {
if let Some(d) = dst {
self.regs.insert(d, VMValue::Void);
}
Ok(())
}
Err(e) => Err(VMError::InvalidInstruction(format!(
"BoxCall {}.{} failed: {:?}",
p.box_type, method, e
))),
}
} else {
// Special-case: minimal runtime fallback for common InstanceBox methods when
// lowered functions are not available (dev robustness). Keeps behavior stable
// without changing semantics in the normal path.
if let Some(inst) = recv_box.as_any().downcast_ref::<crate::instance_v2::InstanceBox>() {
// Generic current() fallback: if object has integer 'position' and string 'text',
// return one character at that position (or empty at EOF). This covers JsonScanner
// and compatible scanners without relying on class name.
if method == "current" && args.is_empty() {
if let Some(crate::value::NyashValue::Integer(pos)) = inst.get_field_ng("position") {
if let Some(crate::value::NyashValue::String(text)) = inst.get_field_ng("text") {
let s = if pos < 0 || (pos as usize) >= text.len() { String::new() } else {
let bytes = text.as_bytes();
let i = pos as usize;
let j = (i + 1).min(bytes.len());
String::from_utf8(bytes[i..j].to_vec()).unwrap_or_default()
};
if let Some(d) = dst { self.regs.insert(d, VMValue::String(s)); }
return Ok(());
}
}
}
}
// Generic toString fallback for any non-plugin box
if method == "toString" {
if let Some(d) = dst {
// Map VoidBox.toString → "null" for JSON-friendly semantics
let s = if recv_box.as_any().downcast_ref::<crate::box_trait::VoidBox>().is_some() {
"null".to_string()
} else {
recv_box.to_string_box().value
};
self.regs.insert(d, VMValue::String(s));
}
return Ok(());
}
// Minimal runtime fallback for common InstanceBox.is_eof when lowered function is not present.
// This avoids cross-class leaks and hard errors in union-like flows.
if method == "is_eof" && args.is_empty() {
if let Some(inst) = recv_box.as_any().downcast_ref::<crate::instance_v2::InstanceBox>() {
if inst.class_name == "JsonToken" {
let is = match inst.get_field_ng("type") {
Some(crate::value::NyashValue::String(ref s)) => s == "EOF",
_ => false,
};
if let Some(d) = dst { self.regs.insert(d, VMValue::Bool(is)); }
return Ok(());
}
if inst.class_name == "JsonScanner" {
let pos = match inst.get_field_ng("position") { Some(crate::value::NyashValue::Integer(i)) => i, _ => 0 };
let len = match inst.get_field_ng("length") { Some(crate::value::NyashValue::Integer(i)) => i, _ => 0 };
let is = pos >= len;
if let Some(d) = dst { self.regs.insert(d, VMValue::Bool(is)); }
return Ok(());
}
}
}
// Dynamic fallback for user-defined InstanceBox: dispatch to lowered function "Class.method/Arity"
if let Some(inst) = recv_box.as_any().downcast_ref::<crate::instance_v2::InstanceBox>() {
let class_name = inst.class_name.clone();
let arity = args.len(); // function name arity excludes 'me'
let fname = format!("{}.{}{}", class_name, method, format!("/{}", arity));
if let Some(func) = self.functions.get(&fname).cloned() {
let mut argv: Vec<VMValue> = Vec::with_capacity(arity + 1);
// Pass receiver as first arg ('me')
argv.push(recv.clone());
for a in args {
argv.push(self.reg_load(*a)?);
}
let ret = self.exec_function_inner(&func, Some(&argv))?;
if let Some(d) = dst { self.regs.insert(d, ret); }
return Ok(());
}
}
// Last-resort dev fallback: tolerate InstanceBox.current() by returning empty string
// when no class-specific handler is available. This avoids hard stops in JSON lint smokes
// while builder rewrite and instance dispatch stabilize.
if method == "current" && args.is_empty() {
if let Some(d) = dst { self.regs.insert(d, VMValue::String(String::new())); }
return Ok(());
}
// VoidBox graceful handling for common container-like methods
// Treat null.receiver.* as safe no-ops that return null/0 where appropriate
if recv_box.type_name() == "VoidBox" {
match method {
"object_get" | "array_get" | "toString" => {
if let Some(d) = dst { self.regs.insert(d, VMValue::Void); }
return Ok(());
}
"stringify" => {
if let Some(d) = dst { self.regs.insert(d, VMValue::String("null".to_string())); }
return Ok(());
}
"array_size" | "length" | "size" => {
if let Some(d) = dst { self.regs.insert(d, VMValue::Integer(0)); }
return Ok(());
}
"object_set" | "array_push" | "set" => {
// No-op setters on null receiver
if let Some(d) = dst { self.regs.insert(d, VMValue::Void); }
return Ok(());
}
_ => {}
}
}
Err(VMError::InvalidInstruction(format!(
"BoxCall unsupported on {}.{}",
recv_box.type_name(),
method
)))
}
}
}