use php_ast::ast::{AssignOp, BinaryOp, ExprKind, UnaryPrefixOp};
use mir_codebase::storage::AssertionKind;
use mir_codebase::Codebase;
use mir_types::{Atomic, Union};
use crate::context::Context;
use crate::db::MirDatabase;
pub fn narrow_from_condition<'arena, 'src>(
expr: &php_ast::ast::Expr<'arena, 'src>,
ctx: &mut Context,
is_true: bool,
codebase: &Codebase,
db: &dyn MirDatabase,
file: &str,
) {
match &expr.kind {
ExprKind::Parenthesized(inner) => {
narrow_from_condition(inner, ctx, is_true, codebase, db, file);
}
ExprKind::UnaryPrefix(u) if u.op == UnaryPrefixOp::BooleanNot => {
narrow_from_condition(u.operand, ctx, !is_true, codebase, db, file);
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanAnd || b.op == BinaryOp::LogicalAnd => {
if is_true {
narrow_from_condition(b.left, ctx, true, codebase, db, file);
narrow_from_condition(b.right, ctx, true, codebase, db, file);
}
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanOr || b.op == BinaryOp::LogicalOr => {
if !is_true {
narrow_from_condition(b.left, ctx, false, codebase, db, file);
narrow_from_condition(b.right, ctx, false, codebase, db, file);
} else {
narrow_or_instanceof_true(b.left, b.right, ctx, codebase, db, file);
}
}
ExprKind::Binary(b) if b.op == BinaryOp::Identical || b.op == BinaryOp::NotIdentical => {
let is_identical = b.op == BinaryOp::Identical;
let effective_true = if is_identical { is_true } else { !is_true };
if matches!(b.right.kind, ExprKind::Null) {
if let Some(name) = extract_var_name(b.left) {
narrow_var_null(ctx, &name, effective_true);
}
} else if matches!(b.left.kind, ExprKind::Null) {
if let Some(name) = extract_var_name(b.right) {
narrow_var_null(ctx, &name, effective_true);
}
}
else if matches!(b.right.kind, ExprKind::Bool(true)) {
if let Some(name) = extract_var_name(b.left) {
narrow_var_bool(ctx, &name, true, effective_true);
}
} else if matches!(b.right.kind, ExprKind::Bool(false)) {
if let Some(name) = extract_var_name(b.left) {
narrow_var_bool(ctx, &name, false, effective_true);
}
}
else if let ExprKind::String(s) = &b.right.kind {
if let Some(name) = extract_var_name(b.left) {
narrow_var_literal_string(ctx, &name, s, effective_true);
}
} else if let ExprKind::String(s) = &b.left.kind {
if let Some(name) = extract_var_name(b.right) {
narrow_var_literal_string(ctx, &name, s, effective_true);
}
}
else if let ExprKind::Int(n) = &b.right.kind {
if let Some(name) = extract_var_name(b.left) {
narrow_var_literal_int(ctx, &name, *n, effective_true);
}
} else if let ExprKind::Int(n) = &b.left.kind {
if let Some(name) = extract_var_name(b.right) {
narrow_var_literal_int(ctx, &name, *n, effective_true);
}
}
}
ExprKind::Binary(b) if b.op == BinaryOp::Equal || b.op == BinaryOp::NotEqual => {
let is_equal = b.op == BinaryOp::Equal;
let effective_true = if is_equal { is_true } else { !is_true };
if matches!(b.right.kind, ExprKind::Null) {
if let Some(name) = extract_var_name(b.left) {
narrow_var_null(ctx, &name, effective_true);
}
} else if matches!(b.left.kind, ExprKind::Null) {
if let Some(name) = extract_var_name(b.right) {
narrow_var_null(ctx, &name, effective_true);
}
}
}
ExprKind::Binary(b) if b.op == BinaryOp::Instanceof => {
let (lhs, extra_negation) = match &b.left.kind {
ExprKind::UnaryPrefix(u) if u.op == UnaryPrefixOp::BooleanNot => (u.operand, true),
ExprKind::Parenthesized(inner) => match &inner.kind {
ExprKind::UnaryPrefix(u) if u.op == UnaryPrefixOp::BooleanNot => {
(u.operand, true)
}
_ => (b.left, false),
},
_ => (b.left, false),
};
let effective_is_true = if extra_negation { !is_true } else { is_true };
if let Some(var_name) = extract_var_name(lhs) {
if let Some(raw_name) = extract_class_name(b.right, ctx.self_fqcn.as_deref()) {
let class_name = codebase.resolve_class_name(file, &raw_name);
let current = ctx.get_var(&var_name);
let narrowed = if effective_is_true {
narrow_instanceof_preserving_subtypes(¤t, &class_name, db)
} else {
filter_out_instanceof_match(¤t, &class_name, db)
};
set_narrowed(ctx, &var_name, ¤t, narrowed, true);
}
}
}
ExprKind::FunctionCall(call) => {
let fn_name_opt: Option<&str> = match &call.name.kind {
ExprKind::Identifier(name) => Some(name),
ExprKind::Variable(name) => Some(name.as_ref()),
_ => None,
};
if let Some(fn_name) = fn_name_opt {
if fn_name.eq_ignore_ascii_case("assert") {
if let Some(arg_expr) = call.args.first() {
narrow_from_condition(&arg_expr.value, ctx, is_true, codebase, db, file);
}
} else if apply_docblock_assertions(call, ctx, is_true, codebase, db, file, fn_name)
{
} else if let Some(arg_expr) = call.args.first() {
if let Some(var_name) = extract_var_name(&arg_expr.value) {
narrow_from_type_fn(ctx, fn_name, &var_name, is_true);
}
}
}
}
ExprKind::Isset(vars) => {
for var_expr in vars.iter() {
if let Some(var_name) = extract_var_name(var_expr) {
if is_true {
let current = ctx.get_var(&var_name);
ctx.set_var(&var_name, current.remove_null());
ctx.assigned_vars.insert(var_name);
}
}
}
}
ExprKind::Assign(a) if matches!(a.op, AssignOp::Assign | AssignOp::Coalesce) => {
if let Some(var_name) = extract_var_name(a.target) {
let current = ctx.get_var(&var_name);
let narrowed = if is_true {
current.narrow_to_truthy()
} else {
current.narrow_to_falsy()
};
if !narrowed.is_empty() {
ctx.set_var(&var_name, narrowed);
} else if !current.is_empty() && !current.is_mixed() {
ctx.diverges = true;
}
}
}
_ => {
if let Some(var_name) = extract_var_name(expr) {
let current = ctx.get_var(&var_name);
let narrowed = if is_true {
current.narrow_to_truthy()
} else {
current.narrow_to_falsy()
};
if !narrowed.is_empty() {
ctx.set_var(&var_name, narrowed);
} else if !current.is_empty() && !current.is_mixed() {
ctx.diverges = true;
}
}
}
}
}
fn apply_docblock_assertions<'arena, 'src>(
call: &php_ast::ast::FunctionCallExpr<'arena, 'src>,
ctx: &mut Context,
is_true: bool,
codebase: &Codebase,
db: &dyn MirDatabase,
file: &str,
fn_name: &str,
) -> bool {
let fn_name = fn_name
.strip_prefix('\\')
.map(|s| s.to_string())
.unwrap_or_else(|| fn_name.to_string());
let fn_active =
|name: &str| -> bool { db.lookup_function_node(name).is_some_and(|n| n.active(db)) };
let resolved_fn_name = {
let qualified = codebase.resolve_class_name(file, &fn_name);
if fn_active(qualified.as_str()) {
qualified
} else if fn_active(fn_name.as_str()) {
fn_name.clone()
} else {
qualified
}
};
let Some(node) = db
.lookup_function_node(resolved_fn_name.as_str())
.filter(|n| n.active(db))
else {
return false;
};
let expected_kind = if is_true {
AssertionKind::AssertIfTrue
} else {
AssertionKind::AssertIfFalse
};
let assertions = node.assertions(db);
let params = node.params(db);
let mut applied = false;
for assertion in assertions
.iter()
.filter(|a| a.kind == expected_kind || (is_true && a.kind == AssertionKind::Assert))
{
if let Some(index) = params.iter().position(|p| p.name == assertion.param) {
if let Some(arg) = call.args.get(index) {
if let Some(var_name) = extract_var_name(&arg.value) {
ctx.set_var(&var_name, assertion.ty.clone());
applied = true;
}
}
}
}
applied
}
fn narrow_or_instanceof_true<'arena, 'src>(
left: &php_ast::ast::Expr<'arena, 'src>,
right: &php_ast::ast::Expr<'arena, 'src>,
ctx: &mut Context,
codebase: &Codebase,
db: &dyn MirDatabase,
file: &str,
) {
let self_fqcn = ctx.self_fqcn.as_deref();
let mut var_name: Option<String> = None;
let mut class_names: Vec<String> = vec![];
fn collect_instanceof<'a, 's>(
expr: &php_ast::ast::Expr<'a, 's>,
var_name: &mut Option<String>,
class_names: &mut Vec<String>,
codebase: &Codebase,
file: &str,
self_fqcn: Option<&str>,
) -> bool {
match &expr.kind {
ExprKind::Binary(b) if b.op == BinaryOp::Instanceof => {
if let (Some(vn), Some(cn)) = (
extract_var_name(b.left),
extract_class_name(b.right, self_fqcn),
) {
let resolved = codebase.resolve_class_name(file, &cn);
match var_name {
None => {
*var_name = Some(vn);
class_names.push(resolved);
true
}
Some(existing) if existing == &vn => {
class_names.push(resolved);
true
}
_ => false, }
} else {
false
}
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanOr || b.op == BinaryOp::LogicalOr => {
collect_instanceof(b.left, var_name, class_names, codebase, file, self_fqcn)
&& collect_instanceof(b.right, var_name, class_names, codebase, file, self_fqcn)
}
ExprKind::Parenthesized(inner) => {
collect_instanceof(inner, var_name, class_names, codebase, file, self_fqcn)
}
_ => false,
}
}
let left_ok = collect_instanceof(
left,
&mut var_name,
&mut class_names,
codebase,
file,
self_fqcn,
);
let right_ok = collect_instanceof(
right,
&mut var_name,
&mut class_names,
codebase,
file,
self_fqcn,
);
if left_ok && right_ok {
if let Some(vn) = var_name {
if !class_names.is_empty() {
let current = ctx.get_var(&vn);
let mut narrowed = Union::empty();
for cn in &class_names {
let n = narrow_instanceof_preserving_subtypes(¤t, cn, db);
narrowed = Union::merge(&narrowed, &n);
}
let result = if narrowed.is_empty() {
current.clone()
} else {
narrowed
};
if !result.is_empty() {
ctx.set_var(&vn, result);
}
}
}
}
}
fn narrow_instanceof_preserving_subtypes(
current: &Union,
class_name: &str,
db: &dyn MirDatabase,
) -> Union {
let narrowed_ty = Atomic::TNamedObject {
fqcn: class_name.into(),
type_params: vec![],
};
if current.is_empty() || current.is_mixed() {
return Union::single(narrowed_ty);
}
let mut result = Union::empty();
result.possibly_undefined = current.possibly_undefined;
result.from_docblock = current.from_docblock;
for atomic in ¤t.types {
match atomic {
Atomic::TNamedObject { fqcn, .. }
| Atomic::TSelf { fqcn }
| Atomic::TStaticObject { fqcn }
| Atomic::TParent { fqcn }
if named_object_matches_instanceof(fqcn, class_name, db) =>
{
result.add_type(atomic.clone());
}
Atomic::TObject | Atomic::TMixed => result.add_type(narrowed_ty.clone()),
_ => {}
}
}
if result.is_empty() {
Union::single(narrowed_ty)
} else {
result
}
}
fn filter_out_instanceof_match(current: &Union, class_name: &str, db: &dyn MirDatabase) -> Union {
current.filter(|t| match t {
Atomic::TNamedObject { fqcn, .. }
| Atomic::TSelf { fqcn }
| Atomic::TStaticObject { fqcn }
| Atomic::TParent { fqcn } => !named_object_matches_instanceof(fqcn, class_name, db),
_ => true,
})
}
fn named_object_matches_instanceof(fqcn: &str, class_name: &str, db: &dyn MirDatabase) -> bool {
fqcn == class_name || crate::db::extends_or_implements_via_db(db, fqcn, class_name)
}
fn set_narrowed(
ctx: &mut Context,
name: &str,
current: &Union,
narrowed: Union,
mark_diverges: bool,
) {
if !narrowed.is_empty() {
ctx.set_var(name, narrowed);
} else if mark_diverges && !current.is_empty() && !current.is_mixed() {
ctx.diverges = true;
}
}
fn narrow_var_null(ctx: &mut Context, name: &str, is_null: bool) {
let current = ctx.get_var(name);
let narrowed = if is_null {
current.narrow_to_null()
} else {
current.remove_null()
};
set_narrowed(ctx, name, ¤t, narrowed, true);
}
fn narrow_var_bool(ctx: &mut Context, name: &str, value: bool, is_value: bool) {
let current = ctx.get_var(name);
let narrowed = if is_value {
if value {
current.filter(|t| matches!(t, Atomic::TTrue | Atomic::TBool | Atomic::TMixed))
} else {
current.filter(|t| matches!(t, Atomic::TFalse | Atomic::TBool | Atomic::TMixed))
}
} else if value {
current.filter(|t| !matches!(t, Atomic::TTrue))
} else {
current.filter(|t| !matches!(t, Atomic::TFalse))
};
set_narrowed(ctx, name, ¤t, narrowed, false);
}
fn narrow_from_type_fn(ctx: &mut Context, fn_name: &str, var_name: &str, is_true: bool) {
let current = ctx.get_var(var_name);
let narrowed = match fn_name.to_lowercase().as_str() {
"is_string" => {
if is_true {
current.narrow_to_string()
} else {
current.filter(|t| !t.is_string())
}
}
"is_int" | "is_integer" | "is_long" => {
if is_true {
current.narrow_to_int()
} else {
current.filter(|t| !t.is_int())
}
}
"is_float" | "is_double" | "is_real" => {
if is_true {
current.narrow_to_float()
} else {
current.filter(|t| !matches!(t, Atomic::TFloat | Atomic::TLiteralFloat(..)))
}
}
"is_bool" => {
if is_true {
current.narrow_to_bool()
} else {
current.filter(|t| !matches!(t, Atomic::TBool | Atomic::TTrue | Atomic::TFalse))
}
}
"is_null" => {
if is_true {
current.narrow_to_null()
} else {
current.remove_null()
}
}
"is_array" => {
if is_true {
current.narrow_to_array()
} else {
current.filter(|t| !t.is_array())
}
}
"is_object" => {
if is_true {
current.narrow_to_object()
} else {
current.filter(|t| !t.is_object())
}
}
"is_callable" => {
if is_true {
current.narrow_to_callable()
} else {
current.filter(|t| !t.is_callable())
}
}
"is_numeric" => {
if is_true {
current.filter(|t| {
matches!(
t,
Atomic::TInt
| Atomic::TFloat
| Atomic::TNumeric
| Atomic::TNumericString
| Atomic::TLiteralInt(_)
| Atomic::TMixed
)
})
} else {
current.filter(|t| {
!matches!(
t,
Atomic::TInt
| Atomic::TFloat
| Atomic::TNumeric
| Atomic::TNumericString
| Atomic::TLiteralInt(_)
)
})
}
}
"method_exists" | "property_exists" => {
if is_true {
Union::single(Atomic::TObject)
} else {
current.clone()
}
}
_ => return,
};
set_narrowed(ctx, var_name, ¤t, narrowed, true);
}
fn narrow_var_literal_string(ctx: &mut Context, name: &str, value: &str, is_value: bool) {
let current = ctx.get_var(name);
let narrowed = if is_value {
current.filter(|t| match t {
Atomic::TLiteralString(s) => s.as_ref() == value,
Atomic::TString | Atomic::TScalar | Atomic::TMixed => true,
_ => false,
})
} else {
current.filter(|t| !matches!(t, Atomic::TLiteralString(s) if s.as_ref() == value))
};
set_narrowed(ctx, name, ¤t, narrowed, false);
}
fn narrow_var_literal_int(ctx: &mut Context, name: &str, value: i64, is_value: bool) {
let current = ctx.get_var(name);
let narrowed = if is_value {
current.filter(|t| match t {
Atomic::TLiteralInt(n) => *n == value,
Atomic::TInt | Atomic::TScalar | Atomic::TNumeric | Atomic::TMixed => true,
_ => false,
})
} else {
current.filter(|t| !matches!(t, Atomic::TLiteralInt(n) if *n == value))
};
set_narrowed(ctx, name, ¤t, narrowed, false);
}
fn extract_var_name<'a, 'arena, 'src>(
expr: &'a php_ast::ast::Expr<'arena, 'src>,
) -> Option<String> {
match &expr.kind {
ExprKind::Variable(name) => Some(name.as_str().trim_start_matches('$').to_string()),
ExprKind::Parenthesized(inner) => extract_var_name(inner),
_ => None,
}
}
fn extract_class_name<'arena, 'src>(
expr: &php_ast::ast::Expr<'arena, 'src>,
self_fqcn: Option<&str>,
) -> Option<String> {
match &expr.kind {
ExprKind::Identifier(name) => Some(name.to_string()),
ExprKind::Variable(name) if name.as_str().trim_start_matches('$') == "this" => {
self_fqcn.map(|s| s.to_string())
}
ExprKind::Variable(_) => None, _ => None,
}
}
trait UnionNarrowExt {
fn filter<F: Fn(&Atomic) -> bool>(&self, f: F) -> Union;
}
impl UnionNarrowExt for Union {
fn filter<F: Fn(&Atomic) -> bool>(&self, f: F) -> Union {
let mut result = Union::empty();
result.possibly_undefined = self.possibly_undefined;
result.from_docblock = self.from_docblock;
for atomic in &self.types {
if f(atomic) {
result.types.push(atomic.clone());
}
}
result
}
}