use php_ast::ast::{AssignOp, BinaryOp, UnaryPrefixOp};
use php_ast::owned::ExprKind;
use mir_codebase::storage::AssertionKind;
use mir_types::{Atomic, Type};
use crate::db::MirDatabase;
use crate::flow_state::FlowState;
pub fn narrow_from_condition(
expr: &php_ast::owned::Expr,
ctx: &mut FlowState,
is_true: bool,
db: &dyn MirDatabase,
file: &str,
) {
match &expr.kind {
ExprKind::Parenthesized(inner) => {
narrow_from_condition(inner, ctx, is_true, db, file);
}
ExprKind::UnaryPrefix(u) if u.op == UnaryPrefixOp::BooleanNot => {
narrow_from_condition(&u.operand, ctx, !is_true, db, file);
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanAnd || b.op == BinaryOp::LogicalAnd => {
if is_true {
narrow_from_condition(&b.left, ctx, true, db, file);
narrow_from_condition(&b.right, ctx, true, db, file);
promote_assignment_effects(&b.left, ctx);
promote_assignment_effects(&b.right, ctx);
}
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanOr || b.op == BinaryOp::LogicalOr => {
if !is_true {
narrow_from_condition(&b.left, ctx, false, db, file);
narrow_from_condition(&b.right, ctx, false, db, file);
promote_assignment_effects(&b.left, ctx);
promote_assignment_effects(&b.right, ctx);
} else {
narrow_or_instanceof_true(&b.left, &b.right, ctx, db, file);
narrow_or_isset_true(&b.left, &b.right, ctx, 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 let Some(nc) = extract_null_coalesce(&b.left) {
if let Some(var_name) = extract_var_name(&nc.left) {
if !effective_true && same_literal(&nc.right, &b.right) {
let current = ctx.get_var(&var_name);
ctx.set_var(&var_name, current.remove_null());
}
}
} else if let Some(nc) = extract_null_coalesce(&b.right) {
if let Some(var_name) = extract_var_name(&nc.left) {
if !effective_true && same_literal(&nc.right, &b.left) {
let current = ctx.get_var(&var_name);
ctx.set_var(&var_name, current.remove_null());
}
}
}
else 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(class_name_str) = &b.right.kind {
if let Some(obj_var_name) = extract_get_class_arg(&b.left) {
let fqcn = crate::db::resolve_name(db, file, class_name_str.as_ref());
narrow_var_to_specific_class(ctx, &obj_var_name, &fqcn, effective_true);
} else if let Some(name) = extract_var_name(&b.left) {
narrow_var_literal_string(ctx, &name, class_name_str, effective_true);
}
} else if let ExprKind::String(class_name_str) = &b.left.kind {
if let Some(obj_var_name) = extract_get_class_arg(&b.right) {
let fqcn = crate::db::resolve_name(db, file, class_name_str.as_ref());
narrow_var_to_specific_class(ctx, &obj_var_name, &fqcn, effective_true);
} else if let Some(name) = extract_var_name(&b.right) {
narrow_var_literal_string(ctx, &name, class_name_str, 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);
}
}
else if let ExprKind::StaticPropertyAccess(_) = &b.right.kind {
if let Some(var_name) = extract_var_name(&b.left) {
if let Some((enum_fqcn, case_name)) =
extract_enum_case(&b.right, ctx.self_fqcn.as_deref(), db, file)
{
narrow_var_to_literal_enum_case(
ctx,
&var_name,
&enum_fqcn,
&case_name,
effective_true,
);
}
}
} else if let ExprKind::StaticPropertyAccess(_) = &b.left.kind {
if let Some(var_name) = extract_var_name(&b.right) {
if let Some((enum_fqcn, case_name)) =
extract_enum_case(&b.left, ctx.self_fqcn.as_deref(), db, file)
{
narrow_var_to_literal_enum_case(
ctx,
&var_name,
&enum_fqcn,
&case_name,
effective_true,
);
}
}
}
else if let ExprKind::ClassConstAccess(cca) = &b.right.kind {
if let Some(var_name) = extract_var_name(&b.left) {
if let Some(fqcn) =
extract_class_const_fqcn(cca, ctx.self_fqcn.as_deref(), db, file)
{
narrow_var_to_class_string(ctx, &var_name, &fqcn, effective_true);
}
}
} else if let ExprKind::ClassConstAccess(cca) = &b.left.kind {
if let Some(var_name) = extract_var_name(&b.right) {
if let Some(fqcn) =
extract_class_const_fqcn(cca, ctx.self_fqcn.as_deref(), db, file)
{
narrow_var_to_class_string(ctx, &var_name, &fqcn, 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 => {
if let Some(var_name) = extract_var_name(&b.left) {
if let Some(raw_name) = extract_class_name(&b.right, ctx.self_fqcn.as_deref()) {
let class_name = crate::db::resolve_name(db, file, &raw_name);
let current = ctx.get_var(&var_name);
let narrowed = if is_true {
narrow_instanceof_preserving_subtypes(
¤t,
&class_name,
db,
&ctx.template_param_names,
)
} 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.as_ref()),
ExprKind::Variable(name) => Some(name.as_ref()),
_ => None,
};
if let Some(fn_name) = fn_name_opt {
let bare = fn_name.trim_start_matches('\\');
if matches!(bare, "class_exists" | "interface_exists" | "trait_exists") {
if is_true {
if let Some(arg_expr) = call.args.first() {
if let Some(fqcn) =
extract_class_fqcn_from_expr(&arg_expr.value, db, file)
{
ctx.class_exists_guards.insert(fqcn);
}
}
}
} else 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, db, file);
}
} else if apply_docblock_assertions(call, ctx, is_true, 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());
std::sync::Arc::make_mut(&mut ctx.assigned_vars)
.insert(mir_types::Name::from(var_name.as_str()));
}
}
}
}
ExprKind::Empty(var_expr) => {
if let Some(var_name) = extract_var_name(var_expr) {
let current = ctx.get_var(&var_name);
let narrowed = if is_true {
current.narrow_to_falsy()
} else {
current.narrow_to_truthy()
};
if !narrowed.is_empty() {
ctx.set_var(&var_name, narrowed);
}
}
}
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(
call: &php_ast::owned::FunctionCallExpr,
ctx: &mut FlowState,
is_true: bool,
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 {
let here = crate::db::Fqcn::from_str(db, name);
crate::db::find_function(db, here).is_some()
};
let resolved_fn_name = {
let qualified = crate::db::resolve_name(db, file, &fn_name);
if fn_active(qualified.as_str()) {
qualified
} else if fn_active(fn_name.as_str()) {
fn_name.clone()
} else {
qualified
}
};
let here = crate::db::Fqcn::from_str(db, resolved_fn_name.as_str());
let Some(f) = crate::db::find_function(db, here) else {
return false;
};
let expected_kind = if is_true {
AssertionKind::AssertIfTrue
} else {
AssertionKind::AssertIfFalse
};
let assertions = &f.assertions;
let params = &f.params;
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(
left: &php_ast::owned::Expr,
right: &php_ast::owned::Expr,
ctx: &mut FlowState,
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(
expr: &php_ast::owned::Expr,
var_name: &mut Option<String>,
class_names: &mut Vec<String>,
db: &dyn MirDatabase,
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 = crate::db::resolve_name(db, 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, db, file, self_fqcn)
&& collect_instanceof(&b.right, var_name, class_names, db, file, self_fqcn)
}
ExprKind::Parenthesized(inner) => {
collect_instanceof(inner, var_name, class_names, db, file, self_fqcn)
}
_ => false,
}
}
let left_ok = collect_instanceof(left, &mut var_name, &mut class_names, db, file, self_fqcn);
let right_ok = collect_instanceof(right, &mut var_name, &mut class_names, db, 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 = Type::empty();
for cn in &class_names {
let n = narrow_instanceof_preserving_subtypes(
¤t,
cn,
db,
&ctx.template_param_names,
);
narrowed.merge_with(&n);
}
let result = if narrowed.is_empty() {
current.clone()
} else {
narrowed
};
if !result.is_empty() {
ctx.set_var(&vn, result);
}
}
}
}
}
fn narrow_or_isset_true(
left: &php_ast::owned::Expr,
right: &php_ast::owned::Expr,
ctx: &mut FlowState,
db: &dyn MirDatabase,
file: &str,
) {
if let ExprKind::UnaryPrefix(u) = &left.kind {
if u.op == UnaryPrefixOp::BooleanNot {
if let ExprKind::Isset(vars) = &u.operand.kind {
let original_vars: Vec<_> = vars
.iter()
.filter_map(|var_expr| {
extract_var_name(var_expr).map(|name| {
let was_assigned = ctx.var_is_defined(&name);
(name.clone(), ctx.get_var(&name), was_assigned)
})
})
.collect();
for var_expr in vars.iter() {
if let Some(var_name) = extract_var_name(var_expr) {
let current = ctx.get_var(&var_name);
ctx.set_var(&var_name, current.remove_null());
std::sync::Arc::make_mut(&mut ctx.assigned_vars)
.insert(mir_types::Name::from(var_name.as_str()));
}
}
narrow_from_condition(right, ctx, true, db, file);
for (var_name, original_type, was_assigned) in original_vars {
let sym = mir_types::Name::from(var_name.as_str());
std::sync::Arc::make_mut(&mut ctx.vars)
.insert(sym, std::sync::Arc::new(original_type));
if !was_assigned {
std::sync::Arc::make_mut(&mut ctx.assigned_vars).remove(&sym);
}
}
}
}
}
}
fn narrow_instanceof_preserving_subtypes(
current: &Type,
class_name: &str,
db: &dyn MirDatabase,
template_param_names: &rustc_hash::FxHashSet<mir_types::Name>,
) -> Type {
let narrowed_ty = Atomic::TNamedObject {
fqcn: class_name.into(),
type_params: mir_types::union::empty_type_params(),
};
if current.is_empty() || current.is_mixed() {
return Type::single(narrowed_ty);
}
let mut result = Type::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::TNamedObject { fqcn, type_params }
if type_params.is_empty()
&& !fqcn.contains('\\')
&& template_param_names.contains(fqcn) =>
{
result.add_type(narrowed_ty.clone());
}
Atomic::TTemplateParam { .. } => {
result.add_type(narrowed_ty.clone());
}
Atomic::TObject | Atomic::TMixed => result.add_type(narrowed_ty.clone()),
_ => {}
}
}
if result.is_empty() {
Type::single(narrowed_ty)
} else {
result
}
}
fn filter_out_instanceof_match(current: &Type, class_name: &str, db: &dyn MirDatabase) -> Type {
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(db, fqcn, class_name)
}
fn set_narrowed(
ctx: &mut FlowState,
name: &str,
current: &Type,
narrowed: Type,
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 FlowState, 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 FlowState, 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 FlowState, 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_scalar" => {
if is_true {
current.narrow_to_scalar()
} else {
current.filter(|t| {
!matches!(
t,
Atomic::TString
| Atomic::TLiteralString(..)
| Atomic::TNumericString
| Atomic::TInt
| Atomic::TLiteralInt(..)
| Atomic::TFloat
| Atomic::TLiteralFloat(..)
| Atomic::TBool
| Atomic::TTrue
| Atomic::TFalse
| Atomic::TScalar
)
})
}
}
"is_iterable" => {
if is_true {
current.narrow_to_iterable()
} else {
current.filter(|t| !t.is_array() && !t.is_object())
}
}
"is_countable" => {
if is_true {
current.narrow_to_countable()
} else {
current.filter(|t| !t.is_array() && !t.is_object())
}
}
"is_resource" => {
if is_true {
current.narrow_to_resource()
} else {
current.clone()
}
}
"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 {
Type::single(Atomic::TObject)
} else {
current.clone()
}
}
_ => return,
};
set_narrowed(ctx, var_name, ¤t, narrowed, true);
}
fn narrow_var_literal_string(ctx: &mut FlowState, 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 FlowState, 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 narrow_var_to_literal_enum_case(
ctx: &mut FlowState,
name: &str,
enum_fqcn: &str,
case_name: &str,
is_case: bool,
) {
let current = ctx.get_var(name);
let narrowed = if is_case {
Type::single(Atomic::TLiteralEnumCase {
enum_fqcn: enum_fqcn.into(),
case_name: case_name.into(),
})
} else {
current.filter(|t| {
!matches!(t, Atomic::TLiteralEnumCase { enum_fqcn: fqcn, case_name: c }
if fqcn.as_ref() == enum_fqcn && c.as_ref() == case_name)
})
};
set_narrowed(ctx, name, ¤t, narrowed, true);
}
fn narrow_var_to_class_string(ctx: &mut FlowState, name: &str, fqcn: &str, is_class: bool) {
let current = ctx.get_var(name);
let narrowed = if is_class {
Type::single(Atomic::TClassString(Some(mir_types::Name::from(fqcn))))
} else {
current.filter(|t| !matches!(t, Atomic::TClassString(Some(f)) if f.as_ref() == fqcn))
};
set_narrowed(ctx, name, ¤t, narrowed, true);
}
fn narrow_var_to_specific_class(ctx: &mut FlowState, name: &str, fqcn: &str, is_exact_class: bool) {
let current = ctx.get_var(name);
let narrowed = if is_exact_class {
Type::single(Atomic::TNamedObject {
fqcn: fqcn.into(),
type_params: mir_types::union::empty_type_params(),
})
} else {
current.filter(|t| match t {
Atomic::TNamedObject { fqcn: obj_fqcn, .. } => obj_fqcn.as_ref() != fqcn,
_ => true,
})
};
set_narrowed(ctx, name, ¤t, narrowed, true);
}
fn extract_class_fqcn_from_expr(
expr: &php_ast::owned::Expr,
db: &dyn MirDatabase,
file: &str,
) -> Option<std::sync::Arc<str>> {
let expr = peel_parens(expr);
match &expr.kind {
ExprKind::ClassConstAccess(cca) => {
if let ExprKind::Identifier(id) = &cca.class.kind {
let member = match &cca.member.kind {
ExprKind::Identifier(s) => s.as_ref(),
_ => return None,
};
if member.eq_ignore_ascii_case("class") {
let resolved = crate::db::resolve_name(db, file, id.as_ref());
if !matches!(resolved.as_str(), "self" | "static" | "parent") {
return Some(std::sync::Arc::from(resolved.as_str()));
}
}
}
None
}
ExprKind::String(s) => {
let name = s.as_ref().trim_start_matches('\\');
if !name.is_empty() {
Some(std::sync::Arc::from(name))
} else {
None
}
}
_ => None,
}
}
fn extract_var_name(expr: &php_ast::owned::Expr) -> Option<String> {
match &expr.kind {
ExprKind::Variable(name) => Some(name.trim_start_matches('$').to_string()),
ExprKind::Parenthesized(inner) => extract_var_name(inner),
_ => None,
}
}
fn extract_null_coalesce(expr: &php_ast::owned::Expr) -> Option<&php_ast::owned::NullCoalesceExpr> {
match &expr.kind {
ExprKind::NullCoalesce(nc) => Some(nc),
ExprKind::Parenthesized(inner) => extract_null_coalesce(inner),
_ => None,
}
}
fn same_literal(a: &php_ast::owned::Expr, b: &php_ast::owned::Expr) -> bool {
let a = peel_parens(a);
let b = peel_parens(b);
match (&a.kind, &b.kind) {
(ExprKind::Null, ExprKind::Null) => true,
(ExprKind::Bool(a), ExprKind::Bool(b)) => a == b,
(ExprKind::Int(a), ExprKind::Int(b)) => a == b,
(ExprKind::String(a), ExprKind::String(b)) => a == b,
_ => false,
}
}
fn peel_parens(expr: &php_ast::owned::Expr) -> &php_ast::owned::Expr {
match &expr.kind {
ExprKind::Parenthesized(inner) => peel_parens(inner),
_ => expr,
}
}
fn extract_class_name(expr: &php_ast::owned::Expr, self_fqcn: Option<&str>) -> Option<String> {
match &expr.kind {
ExprKind::Identifier(name) => Some(name.to_string()),
ExprKind::Variable(name) if name.trim_start_matches('$') == "this" => {
self_fqcn.map(|s| s.to_string())
}
ExprKind::Variable(_) => None, _ => None,
}
}
fn extract_enum_case(
expr: &php_ast::owned::Expr,
self_fqcn: Option<&str>,
db: &dyn MirDatabase,
file: &str,
) -> Option<(String, String)> {
if let ExprKind::StaticPropertyAccess(spa) = &expr.kind {
if let Some(enum_short_name) = extract_class_name(&spa.class, self_fqcn) {
let enum_fqcn = crate::db::resolve_name(db, file, &enum_short_name);
if let ExprKind::Identifier(case_name) = &spa.member.kind {
return Some((enum_fqcn, case_name.to_string()));
}
}
}
None
}
fn extract_class_const_fqcn(
cca: &php_ast::owned::StaticAccessExpr,
self_fqcn: Option<&str>,
db: &dyn MirDatabase,
file: &str,
) -> Option<String> {
let is_class = matches!(&cca.member.kind, ExprKind::Identifier(n) if n.as_ref() == "class");
if !is_class {
return None;
}
let short = extract_class_name(&cca.class, self_fqcn)?;
Some(crate::db::resolve_name(db, file, &short))
}
fn promote_assignment_effects(expr: &php_ast::owned::Expr, ctx: &mut FlowState) {
match &expr.kind {
ExprKind::Assign(a) => {
if let Some(var_name) = extract_var_name(&a.target) {
let sym = mir_types::Name::from(var_name.as_str());
if ctx.possibly_assigned_vars.contains(&sym) {
let ty = ctx.get_var(&var_name);
ctx.set_var(&var_name, ty);
std::sync::Arc::make_mut(&mut ctx.possibly_assigned_vars).remove(&sym);
}
}
promote_assignment_effects(&a.value, ctx);
}
ExprKind::UnaryPrefix(u) => {
promote_assignment_effects(&u.operand, ctx);
}
ExprKind::FunctionCall(call) => {
for arg in call.args.iter() {
promote_assignment_effects(&arg.value, ctx);
}
}
ExprKind::MethodCall(mc) | ExprKind::NullsafeMethodCall(mc) => {
promote_assignment_effects(&mc.object, ctx);
for arg in mc.args.iter() {
promote_assignment_effects(&arg.value, ctx);
}
}
ExprKind::StaticMethodCall(smc) => {
for arg in smc.args.iter() {
promote_assignment_effects(&arg.value, ctx);
}
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanAnd || b.op == BinaryOp::LogicalAnd => {
promote_assignment_effects(&b.left, ctx);
}
ExprKind::Binary(b) if b.op == BinaryOp::BooleanOr || b.op == BinaryOp::LogicalOr => {
promote_assignment_effects(&b.left, ctx);
}
ExprKind::Binary(b) => {
promote_assignment_effects(&b.left, ctx);
promote_assignment_effects(&b.right, ctx);
}
ExprKind::Parenthesized(inner) => {
promote_assignment_effects(inner, ctx);
}
_ => {}
}
}
fn extract_get_class_arg(expr: &php_ast::owned::Expr) -> Option<String> {
if let ExprKind::FunctionCall(call) = &expr.kind {
if let ExprKind::Identifier(name) = &call.name.kind {
if name.eq_ignore_ascii_case("get_class") {
if let Some(arg) = call.args.first() {
return extract_var_name(&arg.value);
}
}
}
}
None
}
trait UnionNarrowExt {
fn filter<F: Fn(&Atomic) -> bool>(&self, f: F) -> Type;
}
impl UnionNarrowExt for Type {
fn filter<F: Fn(&Atomic) -> bool>(&self, f: F) -> Type {
let mut result = Type::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
}
}