When executing ZEND_FETCH_OBJ_IS on a non-object, the handler dispatches to
zend_fetch_property_address_read_helper with type==BP_VAR_IS and ends up
generating an error (isset() is supposed to be quiet) here:
if (Z_TYPE_P(container) != IS_OBJECT ||
!Z_OBJ_HT_P(container)->read_property) {
zend_error(E_NOTICE, "Trying to get property of non-object");
*retval = EG(uninitialized_zval_ptr);
SELECTIVE_PZVAL_LOCK(*retval, &opline->result);
AI_USE_PTR(EX_T(opline->result.u.var).var);
It's cousin ZEND_FETCH_DIM_IS winds up at zend_fetch_dimension_address() where
the non-arrayness of op1 is handled via these case statements:
case IS_NULL: {
/* for read-mode only */
if (result) {
result->var.ptr_ptr = &EG(uninitialized_zval_ptr);
PZVAL_LOCK(*result->var.ptr_ptr);
}
if (type==BP_VAR_W || type==BP_VAR_RW) {
zend_error(E_WARNING, "Cannot use a NULL value as an array");
}
break;
}
default: {
switch (type) {
case BP_VAR_UNSET:
zend_error(E_WARNING, "Cannot unset offset in a
non-array variable");
/* break missing intentionally */
case BP_VAR_R:
case BP_VAR_IS:
retval = &EG(uninitialized_zval_ptr);
break;
default:
retval = &EG(error_zval_ptr);
break;
}
if (result) {
result->var.ptr_ptr = retval;
PZVAL_LOCK(*result->var.ptr_ptr);
}
if (type==BP_VAR_W || type==BP_VAR_RW) {
zend_error(E_WARNING, "Cannot use a scalar value as an
array");
}
}
Which indicate the expected behavior of silent failure. It's probably gone
undetected till now as it requires fetching a property or dimension offset from
a property in a third (non-object) variable within an isset() statement:
$foo = NULL;
isset($foo->bar->baz);
Unless someone can suggest why Objects should be noisy inside isset(), I'd like
to suggest the following fix:
Index: Zend/zend_vm_def.h
===================================================================
RCS file: /repository/ZendEngine2/zend_vm_def.h,v
retrieving revision 1.120
diff -u -r1.120 zend_vm_def.h
--- Zend/zend_vm_def.h 13 Jun 2006 12:56:20 -0000 1.120
+++ Zend/zend_vm_def.h 5 Jul 2006 20:52:41 -0000
@@ -1176,7 +1176,9 @@
if (Z_TYPE_P(container) != IS_OBJECT ||
!Z_OBJ_HT_P(container)->read_property) {
- zend_error(E_NOTICE, "Trying to get property of non-object");
+ if (type != BP_VAR_IS) {
+ zend_error(E_NOTICE, "Trying to get property of
non-object");
+ }
*retval = EG(uninitialized_zval_ptr);
SELECTIVE_PZVAL_LOCK(*retval, &opline->result);
AI_USE_PTR(EX_T(opline->result.u.var).var);
I suggest only excluding BP_VAR_IS since the matching behavior for unset()/DIM
is to throw a notice. I could potentially see excluding BP_VAR_R in order to
match read/DIM, but personally I never liked that behavior :)
-Sara