mod args;
mod callable;
mod function;
mod method;
mod static_call;
pub use args::{check_constructor_args, spread_element_type, CheckArgsParams};
pub(crate) use function::extract_class_docblock;
pub struct CallAnalyzer;
pub(crate) fn consume_arg_assignment(
expr: &php_ast::owned::Expr,
ctx: &mut crate::flow_state::FlowState,
) {
if let php_ast::owned::ExprKind::Assign(a) = &expr.kind {
if let php_ast::owned::ExprKind::Variable(name) = &a.target.kind {
let n = name.trim_start_matches('$');
ctx.read_vars.insert(mir_types::Name::from(n));
ctx.mark_consumed(n);
}
}
}
pub(crate) fn premark_byref_arg_vars(
params: &[mir_codebase::storage::FnParam],
args: &[php_ast::owned::Arg],
ctx: &mut crate::flow_state::FlowState,
) {
use php_ast::owned::ExprKind;
for (i, param) in params.iter().enumerate() {
if !param.is_byref {
continue;
}
let targets: &[php_ast::owned::Arg] = if param.is_variadic {
args.get(i..).unwrap_or(&[])
} else {
args.get(i..=i).unwrap_or(&[])
};
for arg in targets {
if let ExprKind::Variable(name) = &arg.value.kind {
let var_name = name.trim_start_matches('$');
if !ctx.var_is_defined(var_name) {
let ty = param
.ty
.as_ref()
.map(|t| (**t).clone())
.unwrap_or_else(mir_types::Type::mixed);
ctx.set_var(var_name, ty);
}
}
}
}
}
thread_local! {
pub(crate) static ARG_TYPES_BUF: std::cell::RefCell<Option<Vec<mir_types::Type>>> =
const { std::cell::RefCell::new(Some(Vec::new())) };
}